download.go 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. package main
  2. import (
  3. "fmt"
  4. "os"
  5. "os/exec"
  6. "path/filepath"
  7. "strings"
  8. "sync"
  9. "github.com/leonelquinteros/gotext"
  10. "github.com/pkg/errors"
  11. alpm "github.com/Jguer/go-alpm/v2"
  12. "github.com/Jguer/yay/v10/pkg/db"
  13. "github.com/Jguer/yay/v10/pkg/dep"
  14. "github.com/Jguer/yay/v10/pkg/multierror"
  15. "github.com/Jguer/yay/v10/pkg/query"
  16. "github.com/Jguer/yay/v10/pkg/text"
  17. )
  18. const gitDiffRefName = "AUR_SEEN"
  19. // Update the YAY_DIFF_REVIEW ref to HEAD. We use this ref to determine which diff were
  20. // reviewed by the user
  21. func gitUpdateSeenRef(path, name string) error {
  22. _, stderr, err := config.Runtime.CmdRunner.Capture(
  23. config.Runtime.CmdBuilder.BuildGitCmd(
  24. filepath.Join(path, name), "update-ref", gitDiffRefName, "HEAD"), 0)
  25. if err != nil {
  26. return fmt.Errorf("%s %s", stderr, err)
  27. }
  28. return nil
  29. }
  30. // Return wether or not we have reviewed a diff yet. It checks for the existence of
  31. // YAY_DIFF_REVIEW in the git ref-list
  32. func gitHasLastSeenRef(path, name string) bool {
  33. _, _, err := config.Runtime.CmdRunner.Capture(
  34. config.Runtime.CmdBuilder.BuildGitCmd(
  35. filepath.Join(path, name), "rev-parse", "--quiet", "--verify", gitDiffRefName), 0)
  36. return err == nil
  37. }
  38. // Returns the last reviewed hash. If YAY_DIFF_REVIEW exists it will return this hash.
  39. // If it does not it will return empty tree as no diff have been reviewed yet.
  40. func getLastSeenHash(path, name string) (string, error) {
  41. if gitHasLastSeenRef(path, name) {
  42. stdout, stderr, err := config.Runtime.CmdRunner.Capture(
  43. config.Runtime.CmdBuilder.BuildGitCmd(
  44. filepath.Join(path, name), "rev-parse", gitDiffRefName), 0)
  45. if err != nil {
  46. return "", fmt.Errorf("%s %s", stderr, err)
  47. }
  48. lines := strings.Split(stdout, "\n")
  49. return lines[0], nil
  50. }
  51. return gitEmptyTree, nil
  52. }
  53. // Check whether or not a diff exists between the last reviewed diff and
  54. // HEAD@{upstream}
  55. func gitHasDiff(path, name string) (bool, error) {
  56. if gitHasLastSeenRef(path, name) {
  57. stdout, stderr, err := config.Runtime.CmdRunner.Capture(
  58. config.Runtime.CmdBuilder.BuildGitCmd(filepath.Join(path, name), "rev-parse", gitDiffRefName, "HEAD@{upstream}"), 0)
  59. if err != nil {
  60. return false, fmt.Errorf("%s%s", stderr, err)
  61. }
  62. lines := strings.Split(stdout, "\n")
  63. lastseen := lines[0]
  64. upstream := lines[1]
  65. return lastseen != upstream, nil
  66. }
  67. // If YAY_DIFF_REVIEW does not exists, we have never reviewed a diff for this package
  68. // and should display it.
  69. return true, nil
  70. }
  71. // TODO: yay-next passes args through the header, use that to unify ABS and AUR
  72. func gitDownloadABS(url, path, name string) (bool, error) {
  73. if err := os.MkdirAll(path, 0o755); err != nil {
  74. return false, err
  75. }
  76. if _, errExist := os.Stat(filepath.Join(path, name)); os.IsNotExist(errExist) {
  77. cmd := config.Runtime.CmdBuilder.BuildGitCmd(path, "clone", "--no-progress", "--single-branch",
  78. "-b", "packages/"+name, url, name)
  79. _, stderr, err := config.Runtime.CmdRunner.Capture(cmd, 0)
  80. if err != nil {
  81. return false, fmt.Errorf(gotext.Get("error cloning %s: %s", name, stderr))
  82. }
  83. return true, nil
  84. } else if errExist != nil {
  85. return false, fmt.Errorf(gotext.Get("error reading %s", filepath.Join(path, name, ".git")))
  86. }
  87. cmd := config.Runtime.CmdBuilder.BuildGitCmd(filepath.Join(path, name), "pull", "--ff-only")
  88. _, stderr, err := config.Runtime.CmdRunner.Capture(cmd, 0)
  89. if err != nil {
  90. return false, fmt.Errorf(gotext.Get("error fetching %s: %s", name, stderr))
  91. }
  92. return true, nil
  93. }
  94. func gitDownload(url, path, name string) (bool, error) {
  95. _, err := os.Stat(filepath.Join(path, name, ".git"))
  96. if os.IsNotExist(err) {
  97. cmd := config.Runtime.CmdBuilder.BuildGitCmd(path, "clone", "--no-progress", url, name)
  98. _, stderr, errCapture := config.Runtime.CmdRunner.Capture(cmd, 0)
  99. if errCapture != nil {
  100. return false, fmt.Errorf(gotext.Get("error cloning %s: %s", name, stderr))
  101. }
  102. return true, nil
  103. } else if err != nil {
  104. return false, fmt.Errorf(gotext.Get("error reading %s", filepath.Join(path, name, ".git")))
  105. }
  106. cmd := config.Runtime.CmdBuilder.BuildGitCmd(filepath.Join(path, name), "fetch")
  107. _, stderr, err := config.Runtime.CmdRunner.Capture(cmd, 0)
  108. if err != nil {
  109. return false, fmt.Errorf(gotext.Get("error fetching %s: %s", name, stderr))
  110. }
  111. return false, nil
  112. }
  113. func gitMerge(path, name string) error {
  114. _, stderr, err := config.Runtime.CmdRunner.Capture(
  115. config.Runtime.CmdBuilder.BuildGitCmd(
  116. filepath.Join(path, name), "reset", "--hard", "HEAD"), 0)
  117. if err != nil {
  118. return fmt.Errorf(gotext.Get("error resetting %s: %s", name, stderr))
  119. }
  120. _, stderr, err = config.Runtime.CmdRunner.Capture(
  121. config.Runtime.CmdBuilder.BuildGitCmd(
  122. filepath.Join(path, name), "merge", "--no-edit", "--ff"), 0)
  123. if err != nil {
  124. return fmt.Errorf(gotext.Get("error merging %s: %s", name, stderr))
  125. }
  126. return nil
  127. }
  128. func getPkgbuilds(pkgs []string, dbExecutor db.Executor, force bool) error {
  129. missing := false
  130. wd, err := os.Getwd()
  131. if err != nil {
  132. return err
  133. }
  134. pkgs = query.RemoveInvalidTargets(pkgs, config.Runtime.Mode)
  135. aur, repo := packageSlices(pkgs, dbExecutor)
  136. for n := range aur {
  137. _, pkg := text.SplitDBFromName(aur[n])
  138. aur[n] = pkg
  139. }
  140. info, err := query.AURInfoPrint(aur, config.RequestSplitN)
  141. if err != nil {
  142. return err
  143. }
  144. if len(repo) > 0 {
  145. missing, err = getPkgbuildsfromABS(repo, wd, dbExecutor, force)
  146. if err != nil {
  147. return err
  148. }
  149. }
  150. if len(aur) > 0 {
  151. allBases := dep.GetBases(info)
  152. bases := make([]dep.Base, 0)
  153. for _, base := range allBases {
  154. name := base.Pkgbase()
  155. pkgDest := filepath.Join(wd, name)
  156. _, err = os.Stat(pkgDest)
  157. if os.IsNotExist(err) {
  158. bases = append(bases, base)
  159. } else if err != nil {
  160. text.Errorln(err)
  161. continue
  162. } else {
  163. if force {
  164. if err = os.RemoveAll(pkgDest); err != nil {
  165. text.Errorln(err)
  166. continue
  167. }
  168. bases = append(bases, base)
  169. } else {
  170. text.Warnln(gotext.Get("%s already exists. Use -f/--force to overwrite", pkgDest))
  171. continue
  172. }
  173. }
  174. }
  175. if _, err = downloadPkgbuilds(bases, nil, wd); err != nil {
  176. return err
  177. }
  178. missing = missing || len(aur) != len(info)
  179. }
  180. if missing {
  181. err = fmt.Errorf("")
  182. }
  183. return err
  184. }
  185. // GetPkgbuild downloads pkgbuild from the ABS.
  186. func getPkgbuildsfromABS(pkgs []string, path string, dbExecutor db.Executor, force bool) (bool, error) {
  187. var wg sync.WaitGroup
  188. var mux sync.Mutex
  189. var errs multierror.MultiError
  190. names := make(map[string]string)
  191. missing := make([]string, 0)
  192. downloaded := 0
  193. for _, pkgN := range pkgs {
  194. var pkg alpm.IPackage
  195. var err error
  196. var url string
  197. pkgDB, name := text.SplitDBFromName(pkgN)
  198. if pkgDB != "" {
  199. pkg = dbExecutor.SatisfierFromDB(name, pkgDB)
  200. } else {
  201. pkg = dbExecutor.SyncSatisfier(name)
  202. }
  203. if pkg == nil {
  204. missing = append(missing, name)
  205. continue
  206. }
  207. name = pkg.Base()
  208. if name == "" {
  209. name = pkg.Name()
  210. }
  211. // TODO: Check existence with ls-remote
  212. // https://git.archlinux.org/svntogit/packages.git
  213. switch pkg.DB().Name() {
  214. case "core", "extra", "testing":
  215. url = "https://git.archlinux.org/svntogit/packages.git"
  216. case "community", "multilib", "community-testing", "multilib-testing":
  217. url = "https://git.archlinux.org/svntogit/community.git"
  218. default:
  219. missing = append(missing, name)
  220. continue
  221. }
  222. _, err = os.Stat(filepath.Join(path, name))
  223. switch {
  224. case err != nil && !os.IsNotExist(err):
  225. text.Errorln(err)
  226. continue
  227. case os.IsNotExist(err), force:
  228. if err = os.RemoveAll(filepath.Join(path, name)); err != nil {
  229. text.Errorln(err)
  230. continue
  231. }
  232. default:
  233. text.Warn(gotext.Get("%s already downloaded -- use -f to overwrite", text.Cyan(name)))
  234. continue
  235. }
  236. names[name] = url
  237. }
  238. if len(missing) != 0 {
  239. text.Warnln(gotext.Get("Missing ABS packages:"),
  240. text.Cyan(strings.Join(missing, ", ")))
  241. }
  242. download := func(pkg string, url string) {
  243. defer wg.Done()
  244. if _, err := gitDownloadABS(url, config.ABSDir, pkg); err != nil {
  245. errs.Add(errors.New(gotext.Get("failed to get pkgbuild: %s: %s", text.Cyan(pkg), err.Error())))
  246. return
  247. }
  248. _, stderr, err := config.Runtime.CmdRunner.Capture(
  249. exec.Command(
  250. "cp", "-r",
  251. filepath.Join(config.ABSDir, pkg, "trunk"),
  252. filepath.Join(path, pkg)), 0)
  253. mux.Lock()
  254. downloaded++
  255. if err != nil {
  256. errs.Add(errors.New(gotext.Get("failed to link %s: %s", text.Cyan(pkg), stderr)))
  257. } else {
  258. fmt.Fprintln(os.Stdout, gotext.Get("(%d/%d) Downloaded PKGBUILD from ABS: %s", downloaded, len(names), text.Cyan(pkg)))
  259. }
  260. mux.Unlock()
  261. }
  262. count := 0
  263. for name, url := range names {
  264. wg.Add(1)
  265. go download(name, url)
  266. count++
  267. if count%25 == 0 {
  268. wg.Wait()
  269. }
  270. }
  271. wg.Wait()
  272. return len(missing) != 0, errs.Return()
  273. }