upgrade.go 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424
  1. package main
  2. import (
  3. "fmt"
  4. "sort"
  5. "strings"
  6. "sync"
  7. "unicode"
  8. alpm "github.com/Jguer/go-alpm"
  9. "github.com/leonelquinteros/gotext"
  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. rpc "github.com/mikkeloscar/aur"
  15. "github.com/Jguer/yay/v10/pkg/multierror"
  16. "github.com/Jguer/yay/v10/pkg/stringset"
  17. )
  18. // upgrade type describes a system upgrade.
  19. type upgrade struct {
  20. Name string
  21. Repository string
  22. LocalVersion string
  23. RemoteVersion string
  24. }
  25. // upSlice is a slice of Upgrades
  26. type upSlice []upgrade
  27. func (u upSlice) Len() int { return len(u) }
  28. func (u upSlice) Swap(i, j int) { u[i], u[j] = u[j], u[i] }
  29. func (u upSlice) Less(i, j int) bool {
  30. if u[i].Repository == u[j].Repository {
  31. iRunes := []rune(u[i].Name)
  32. jRunes := []rune(u[j].Name)
  33. return text.LessRunes(iRunes, jRunes)
  34. }
  35. syncDB, err := config.Runtime.AlpmHandle.SyncDBs()
  36. if err != nil {
  37. iRunes := []rune(u[i].Repository)
  38. jRunes := []rune(u[j].Repository)
  39. return text.LessRunes(iRunes, jRunes)
  40. }
  41. less := false
  42. found := syncDB.ForEach(func(db alpm.DB) error {
  43. switch db.Name() {
  44. case u[i].Repository:
  45. less = true
  46. case u[j].Repository:
  47. less = false
  48. default:
  49. return nil
  50. }
  51. return fmt.Errorf("")
  52. })
  53. if found != nil {
  54. return less
  55. }
  56. iRunes := []rune(u[i].Repository)
  57. jRunes := []rune(u[j].Repository)
  58. return text.LessRunes(iRunes, jRunes)
  59. }
  60. func getVersionDiff(oldVersion, newVersion string) (left, right string) {
  61. if oldVersion == newVersion {
  62. return oldVersion + red(""), newVersion + green("")
  63. }
  64. diffPosition := 0
  65. checkWords := func(str string, index int, words ...string) bool {
  66. for _, word := range words {
  67. wordLength := len(word)
  68. nextIndex := index + 1
  69. if (index < len(str)-wordLength) &&
  70. (str[nextIndex:(nextIndex+wordLength)] == word) {
  71. return true
  72. }
  73. }
  74. return false
  75. }
  76. for index, char := range oldVersion {
  77. charIsSpecial := !(unicode.IsLetter(char) || unicode.IsNumber(char))
  78. if (index >= len(newVersion)) || (char != rune(newVersion[index])) {
  79. if charIsSpecial {
  80. diffPosition = index
  81. }
  82. break
  83. }
  84. if charIsSpecial ||
  85. (((index == len(oldVersion)-1) || (index == len(newVersion)-1)) &&
  86. ((len(oldVersion) != len(newVersion)) ||
  87. (oldVersion[index] == newVersion[index]))) ||
  88. checkWords(oldVersion, index, "rc", "pre", "alpha", "beta") {
  89. diffPosition = index + 1
  90. }
  91. }
  92. samePart := oldVersion[0:diffPosition]
  93. left = samePart + red(oldVersion[diffPosition:])
  94. right = samePart + green(newVersion[diffPosition:])
  95. return left, right
  96. }
  97. // upList returns lists of packages to upgrade from each source.
  98. func upList(warnings *query.AURWarnings, alpmHandle *alpm.Handle, enableDowngrade bool) (aurUp, repoUp upSlice, err error) {
  99. _, remote, _, remoteNames, err := query.FilterPackages(alpmHandle)
  100. if err != nil {
  101. return nil, nil, err
  102. }
  103. var wg sync.WaitGroup
  104. var develUp upSlice
  105. var errs multierror.MultiError
  106. aurdata := make(map[string]*rpc.Pkg)
  107. for _, pkg := range remote {
  108. if pkg.ShouldIgnore() {
  109. warnings.Ignore.Set(pkg.Name())
  110. }
  111. }
  112. if config.Runtime.Mode == settings.ModeAny || config.Runtime.Mode == settings.ModeRepo {
  113. text.OperationInfoln(gotext.Get("Searching databases for updates..."))
  114. wg.Add(1)
  115. go func() {
  116. repoUp, err = upRepo(alpmHandle, enableDowngrade)
  117. errs.Add(err)
  118. wg.Done()
  119. }()
  120. }
  121. if config.Runtime.Mode == settings.ModeAny || config.Runtime.Mode == settings.ModeAUR {
  122. text.OperationInfoln(gotext.Get("Searching AUR for updates..."))
  123. var _aurdata []*rpc.Pkg
  124. _aurdata, err = query.AURInfo(remoteNames, warnings, config.RequestSplitN)
  125. errs.Add(err)
  126. if err == nil {
  127. for _, pkg := range _aurdata {
  128. aurdata[pkg.Name] = pkg
  129. }
  130. wg.Add(1)
  131. go func() {
  132. aurUp = upAUR(remote, aurdata)
  133. wg.Done()
  134. }()
  135. if config.Devel {
  136. text.OperationInfoln(gotext.Get("Checking development packages..."))
  137. wg.Add(1)
  138. go func() {
  139. develUp = upDevel(remote, aurdata)
  140. wg.Done()
  141. }()
  142. }
  143. }
  144. }
  145. wg.Wait()
  146. printLocalNewerThanAUR(remote, aurdata)
  147. if develUp != nil {
  148. names := make(stringset.StringSet)
  149. for _, up := range develUp {
  150. names.Set(up.Name)
  151. }
  152. for _, up := range aurUp {
  153. if !names.Get(up.Name) {
  154. develUp = append(develUp, up)
  155. }
  156. }
  157. aurUp = develUp
  158. }
  159. return aurUp, repoUp, errs.Return()
  160. }
  161. func upDevel(remote []alpm.Package, aurdata map[string]*rpc.Pkg) upSlice {
  162. toUpdate := make([]alpm.Package, 0)
  163. toRemove := make([]string, 0)
  164. var mux1 sync.Mutex
  165. var mux2 sync.Mutex
  166. var wg sync.WaitGroup
  167. checkUpdate := func(vcsName string, e shaInfos) {
  168. defer wg.Done()
  169. if e.needsUpdate() {
  170. if _, ok := aurdata[vcsName]; ok {
  171. for _, pkg := range remote {
  172. if pkg.Name() == vcsName {
  173. mux1.Lock()
  174. toUpdate = append(toUpdate, pkg)
  175. mux1.Unlock()
  176. return
  177. }
  178. }
  179. }
  180. mux2.Lock()
  181. toRemove = append(toRemove, vcsName)
  182. mux2.Unlock()
  183. }
  184. }
  185. for vcsName, e := range savedInfo {
  186. wg.Add(1)
  187. go checkUpdate(vcsName, e)
  188. }
  189. wg.Wait()
  190. toUpgrade := make(upSlice, 0, len(toUpdate))
  191. for _, pkg := range toUpdate {
  192. if pkg.ShouldIgnore() {
  193. printIgnoringPackage(pkg, "latest-commit")
  194. } else {
  195. toUpgrade = append(toUpgrade, upgrade{pkg.Name(), "devel", pkg.Version(), "latest-commit"})
  196. }
  197. }
  198. removeVCSPackage(toRemove)
  199. return toUpgrade
  200. }
  201. // upAUR gathers foreign packages and checks if they have new versions.
  202. // Output: Upgrade type package list.
  203. func upAUR(remote []alpm.Package, aurdata map[string]*rpc.Pkg) upSlice {
  204. toUpgrade := make(upSlice, 0)
  205. for _, pkg := range remote {
  206. aurPkg, ok := aurdata[pkg.Name()]
  207. if !ok {
  208. continue
  209. }
  210. if (config.TimeUpdate && (int64(aurPkg.LastModified) > pkg.BuildDate().Unix())) ||
  211. (alpm.VerCmp(pkg.Version(), aurPkg.Version) < 0) {
  212. if pkg.ShouldIgnore() {
  213. printIgnoringPackage(pkg, aurPkg.Version)
  214. } else {
  215. toUpgrade = append(toUpgrade, upgrade{aurPkg.Name, "aur", pkg.Version(), aurPkg.Version})
  216. }
  217. }
  218. }
  219. return toUpgrade
  220. }
  221. func printIgnoringPackage(pkg alpm.Package, newPkgVersion string) {
  222. left, right := getVersionDiff(pkg.Version(), newPkgVersion)
  223. text.Warnln(gotext.Get("%s: ignoring package upgrade (%s => %s)",
  224. cyan(pkg.Name()),
  225. left, right,
  226. ))
  227. }
  228. func printLocalNewerThanAUR(
  229. remote []alpm.Package, aurdata map[string]*rpc.Pkg) {
  230. for _, pkg := range remote {
  231. aurPkg, ok := aurdata[pkg.Name()]
  232. if !ok {
  233. continue
  234. }
  235. left, right := getVersionDiff(pkg.Version(), aurPkg.Version)
  236. if !isDevelPackage(pkg) && alpm.VerCmp(pkg.Version(), aurPkg.Version) > 0 {
  237. text.Warnln(gotext.Get("%s: local (%s) is newer than AUR (%s)",
  238. cyan(pkg.Name()),
  239. left, right,
  240. ))
  241. }
  242. }
  243. }
  244. func isDevelName(name string) bool {
  245. for _, suffix := range []string{"git", "svn", "hg", "bzr", "nightly"} {
  246. if strings.HasSuffix(name, "-"+suffix) {
  247. return true
  248. }
  249. }
  250. return strings.Contains(name, "-always-")
  251. }
  252. func isDevelPackage(pkg alpm.Package) bool {
  253. return isDevelName(pkg.Name()) || isDevelName(pkg.Base())
  254. }
  255. // upRepo gathers local packages and checks if they have new versions.
  256. // Output: Upgrade type package list.
  257. func upRepo(alpmHandle *alpm.Handle, enableDowngrade bool) (upSlice, error) {
  258. slice := upSlice{}
  259. localDB, err := alpmHandle.LocalDB()
  260. if err != nil {
  261. return slice, err
  262. }
  263. err = alpmHandle.TransInit(alpm.TransFlagNoLock)
  264. if err != nil {
  265. return slice, err
  266. }
  267. defer func() {
  268. err = alpmHandle.TransRelease()
  269. }()
  270. err = alpmHandle.SyncSysupgrade(enableDowngrade)
  271. if err != nil {
  272. return slice, err
  273. }
  274. _ = alpmHandle.TransGetAdd().ForEach(func(pkg alpm.Package) error {
  275. localVer := "-"
  276. if localPkg := localDB.Pkg(pkg.Name()); localPkg != nil {
  277. localVer = localPkg.Version()
  278. }
  279. slice = append(slice, upgrade{
  280. pkg.Name(),
  281. pkg.DB().Name(),
  282. localVer,
  283. pkg.Version(),
  284. })
  285. return nil
  286. })
  287. return slice, nil
  288. }
  289. // upgradePkgs handles updating the cache and installing updates.
  290. func upgradePkgs(aurUp, repoUp upSlice) (ignore, aurNames stringset.StringSet, err error) {
  291. ignore = make(stringset.StringSet)
  292. aurNames = make(stringset.StringSet)
  293. allUpLen := len(repoUp) + len(aurUp)
  294. if allUpLen == 0 {
  295. return ignore, aurNames, nil
  296. }
  297. if !config.UpgradeMenu {
  298. for _, pkg := range aurUp {
  299. aurNames.Set(pkg.Name)
  300. }
  301. return ignore, aurNames, nil
  302. }
  303. sort.Sort(repoUp)
  304. sort.Sort(aurUp)
  305. allUp := append(repoUp, aurUp...)
  306. fmt.Printf("%s"+bold(" %d ")+"%s\n", bold(cyan("::")), allUpLen, bold(gotext.Get("Packages to upgrade.")))
  307. allUp.print()
  308. text.Infoln(gotext.Get("Packages to exclude: (eg: \"1 2 3\", \"1-3\", \"^4\" or repo name)"))
  309. numbers, err := getInput(config.AnswerUpgrade)
  310. if err != nil {
  311. return nil, nil, err
  312. }
  313. // upgrade menu asks you which packages to NOT upgrade so in this case
  314. // include and exclude are kind of swapped
  315. include, exclude, otherInclude, otherExclude := intrange.ParseNumberMenu(numbers)
  316. isInclude := len(exclude) == 0 && len(otherExclude) == 0
  317. for i, pkg := range repoUp {
  318. if isInclude && otherInclude.Get(pkg.Repository) {
  319. ignore.Set(pkg.Name)
  320. }
  321. if isInclude && !include.Get(len(repoUp)-i+len(aurUp)) {
  322. continue
  323. }
  324. if !isInclude && (exclude.Get(len(repoUp)-i+len(aurUp)) || otherExclude.Get(pkg.Repository)) {
  325. continue
  326. }
  327. ignore.Set(pkg.Name)
  328. }
  329. for i, pkg := range aurUp {
  330. if isInclude && otherInclude.Get(pkg.Repository) {
  331. continue
  332. }
  333. if isInclude && !include.Get(len(aurUp)-i) {
  334. aurNames.Set(pkg.Name)
  335. }
  336. if !isInclude && (exclude.Get(len(aurUp)-i) || otherExclude.Get(pkg.Repository)) {
  337. aurNames.Set(pkg.Name)
  338. }
  339. }
  340. return ignore, aurNames, err
  341. }