upgrade.go 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. package main
  2. import (
  3. "fmt"
  4. "sort"
  5. "strings"
  6. "sync"
  7. "github.com/Jguer/go-alpm/v2"
  8. "github.com/leonelquinteros/gotext"
  9. "github.com/Jguer/yay/v10/pkg/db"
  10. "github.com/Jguer/yay/v10/pkg/intrange"
  11. "github.com/Jguer/yay/v10/pkg/query"
  12. "github.com/Jguer/yay/v10/pkg/settings"
  13. "github.com/Jguer/yay/v10/pkg/text"
  14. "github.com/Jguer/yay/v10/pkg/upgrade"
  15. "github.com/Jguer/yay/v10/pkg/vcs"
  16. rpc "github.com/mikkeloscar/aur"
  17. "github.com/Jguer/yay/v10/pkg/multierror"
  18. "github.com/Jguer/yay/v10/pkg/stringset"
  19. )
  20. // upList returns lists of packages to upgrade from each source.
  21. func upList(warnings *query.AURWarnings, dbExecutor db.Executor, enableDowngrade bool) (aurUp, repoUp upgrade.UpSlice, err error) {
  22. remote, remoteNames := query.GetRemotePackages(dbExecutor)
  23. var wg sync.WaitGroup
  24. var develUp upgrade.UpSlice
  25. var errs multierror.MultiError
  26. aurdata := make(map[string]*rpc.Pkg)
  27. for _, pkg := range remote {
  28. if pkg.ShouldIgnore() {
  29. warnings.Ignore.Set(pkg.Name())
  30. }
  31. }
  32. if config.Runtime.Mode == settings.ModeAny || config.Runtime.Mode == settings.ModeRepo {
  33. text.OperationInfoln(gotext.Get("Searching databases for updates..."))
  34. wg.Add(1)
  35. go func() {
  36. repoUp, err = dbExecutor.RepoUpgrades(enableDowngrade)
  37. errs.Add(err)
  38. wg.Done()
  39. }()
  40. }
  41. if config.Runtime.Mode == settings.ModeAny || config.Runtime.Mode == settings.ModeAUR {
  42. text.OperationInfoln(gotext.Get("Searching AUR for updates..."))
  43. var _aurdata []*rpc.Pkg
  44. _aurdata, err = query.AURInfo(remoteNames, warnings, config.RequestSplitN)
  45. errs.Add(err)
  46. if err == nil {
  47. for _, pkg := range _aurdata {
  48. aurdata[pkg.Name] = pkg
  49. }
  50. wg.Add(1)
  51. go func() {
  52. aurUp = upAUR(remote, aurdata, config.TimeUpdate)
  53. wg.Done()
  54. }()
  55. if config.Devel {
  56. text.OperationInfoln(gotext.Get("Checking development packages..."))
  57. wg.Add(1)
  58. go func() {
  59. develUp = upDevel(remote, aurdata, config.Runtime.VCSStore)
  60. wg.Done()
  61. }()
  62. }
  63. }
  64. }
  65. wg.Wait()
  66. printLocalNewerThanAUR(remote, aurdata)
  67. if develUp != nil {
  68. names := make(stringset.StringSet)
  69. for _, up := range develUp {
  70. names.Set(up.Name)
  71. }
  72. for _, up := range aurUp {
  73. if !names.Get(up.Name) {
  74. develUp = append(develUp, up)
  75. }
  76. }
  77. aurUp = develUp
  78. }
  79. return aurUp, repoUp, errs.Return()
  80. }
  81. func upDevel(
  82. remote []db.RepoPackage,
  83. aurdata map[string]*rpc.Pkg,
  84. localCache *vcs.InfoStore) upgrade.UpSlice {
  85. toUpdate := make([]db.RepoPackage, 0, len(aurdata))
  86. toRemove := make([]string, 0)
  87. var mux1, mux2 sync.Mutex
  88. var wg sync.WaitGroup
  89. checkUpdate := func(pkgName string, e vcs.OriginInfoByURL) {
  90. defer wg.Done()
  91. if localCache.NeedsUpdate(e) {
  92. if _, ok := aurdata[pkgName]; ok {
  93. for _, pkg := range remote {
  94. if pkg.Name() == pkgName {
  95. mux1.Lock()
  96. toUpdate = append(toUpdate, pkg)
  97. mux1.Unlock()
  98. return
  99. }
  100. }
  101. }
  102. mux2.Lock()
  103. toRemove = append(toRemove, pkgName)
  104. mux2.Unlock()
  105. }
  106. }
  107. for pkgName, e := range localCache.OriginsByPackage {
  108. wg.Add(1)
  109. go checkUpdate(pkgName, e)
  110. }
  111. wg.Wait()
  112. toUpgrade := make(upgrade.UpSlice, 0, len(toUpdate))
  113. for _, pkg := range toUpdate {
  114. if pkg.ShouldIgnore() {
  115. printIgnoringPackage(pkg, "latest-commit")
  116. } else {
  117. toUpgrade = append(toUpgrade,
  118. upgrade.Upgrade{
  119. Name: pkg.Name(),
  120. Repository: "devel",
  121. LocalVersion: pkg.Version(),
  122. RemoteVersion: "latest-commit",
  123. })
  124. }
  125. }
  126. localCache.RemovePackage(toRemove)
  127. return toUpgrade
  128. }
  129. func printIgnoringPackage(pkg db.RepoPackage, newPkgVersion string) {
  130. left, right := upgrade.GetVersionDiff(pkg.Version(), newPkgVersion)
  131. text.Warnln(gotext.Get("%s: ignoring package upgrade (%s => %s)",
  132. text.Cyan(pkg.Name()),
  133. left, right,
  134. ))
  135. }
  136. // upAUR gathers foreign packages and checks if they have new versions.
  137. // Output: Upgrade type package list.
  138. func upAUR(remote []db.RepoPackage, aurdata map[string]*rpc.Pkg, timeUpdate bool) upgrade.UpSlice {
  139. toUpgrade := make(upgrade.UpSlice, 0)
  140. for _, pkg := range remote {
  141. aurPkg, ok := aurdata[pkg.Name()]
  142. if !ok {
  143. continue
  144. }
  145. if (timeUpdate && (int64(aurPkg.LastModified) > pkg.BuildDate().Unix())) ||
  146. (alpm.VerCmp(pkg.Version(), aurPkg.Version) < 0) {
  147. if pkg.ShouldIgnore() {
  148. printIgnoringPackage(pkg, aurPkg.Version)
  149. } else {
  150. toUpgrade = append(toUpgrade,
  151. upgrade.Upgrade{
  152. Name: aurPkg.Name,
  153. Repository: "aur",
  154. LocalVersion: pkg.Version(),
  155. RemoteVersion: aurPkg.Version,
  156. })
  157. }
  158. }
  159. }
  160. return toUpgrade
  161. }
  162. func printLocalNewerThanAUR(
  163. remote []db.RepoPackage, aurdata map[string]*rpc.Pkg) {
  164. for _, pkg := range remote {
  165. aurPkg, ok := aurdata[pkg.Name()]
  166. if !ok {
  167. continue
  168. }
  169. left, right := upgrade.GetVersionDiff(pkg.Version(), aurPkg.Version)
  170. if !isDevelPackage(pkg) && alpm.VerCmp(pkg.Version(), aurPkg.Version) > 0 {
  171. text.Warnln(gotext.Get("%s: local (%s) is newer than AUR (%s)",
  172. text.Cyan(pkg.Name()),
  173. left, right,
  174. ))
  175. }
  176. }
  177. }
  178. func isDevelName(name string) bool {
  179. for _, suffix := range []string{"git", "svn", "hg", "bzr", "nightly"} {
  180. if strings.HasSuffix(name, "-"+suffix) {
  181. return true
  182. }
  183. }
  184. return strings.Contains(name, "-always-")
  185. }
  186. func isDevelPackage(pkg db.RepoPackage) bool {
  187. return isDevelName(pkg.Name()) || isDevelName(pkg.Base())
  188. }
  189. // upgradePkgs handles updating the cache and installing updates.
  190. func upgradePkgs(aurUp, repoUp upgrade.UpSlice) (ignore, aurNames stringset.StringSet, err error) {
  191. ignore = make(stringset.StringSet)
  192. aurNames = make(stringset.StringSet)
  193. allUpLen := len(repoUp) + len(aurUp)
  194. if allUpLen == 0 {
  195. return ignore, aurNames, nil
  196. }
  197. if !config.UpgradeMenu {
  198. for _, pkg := range aurUp {
  199. aurNames.Set(pkg.Name)
  200. }
  201. return ignore, aurNames, nil
  202. }
  203. sort.Sort(repoUp)
  204. sort.Sort(aurUp)
  205. allUp := append(repoUp, aurUp...)
  206. fmt.Printf("%s"+text.Bold(" %d ")+"%s\n", text.Bold(text.Cyan("::")), allUpLen, text.Bold(gotext.Get("Packages to upgrade.")))
  207. allUp.Print()
  208. text.Infoln(gotext.Get("Packages to exclude: (eg: \"1 2 3\", \"1-3\", \"^4\" or repo name)"))
  209. numbers, err := getInput(config.AnswerUpgrade)
  210. if err != nil {
  211. return nil, nil, err
  212. }
  213. // upgrade menu asks you which packages to NOT upgrade so in this case
  214. // include and exclude are kind of swapped
  215. include, exclude, otherInclude, otherExclude := intrange.ParseNumberMenu(numbers)
  216. isInclude := len(exclude) == 0 && len(otherExclude) == 0
  217. for i, pkg := range repoUp {
  218. if isInclude && otherInclude.Get(pkg.Repository) {
  219. ignore.Set(pkg.Name)
  220. }
  221. if isInclude && !include.Get(len(repoUp)-i+len(aurUp)) {
  222. continue
  223. }
  224. if !isInclude && (exclude.Get(len(repoUp)-i+len(aurUp)) || otherExclude.Get(pkg.Repository)) {
  225. continue
  226. }
  227. ignore.Set(pkg.Name)
  228. }
  229. for i, pkg := range aurUp {
  230. if isInclude && otherInclude.Get(pkg.Repository) {
  231. continue
  232. }
  233. if isInclude && !include.Get(len(aurUp)-i) {
  234. aurNames.Set(pkg.Name)
  235. }
  236. if !isInclude && (exclude.Get(len(aurUp)-i) || otherExclude.Get(pkg.Repository)) {
  237. aurNames.Set(pkg.Name)
  238. }
  239. }
  240. return ignore, aurNames, err
  241. }