upgrade.go 6.9 KB

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