download.go 8.0 KB

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