download.go 8.3 KB

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