upgrade.go 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. package main
  2. import (
  3. "bufio"
  4. "fmt"
  5. "os"
  6. "sort"
  7. "strconv"
  8. "strings"
  9. "unicode"
  10. alpm "github.com/jguer/go-alpm"
  11. rpc "github.com/mikkeloscar/aur"
  12. pkgb "github.com/mikkeloscar/gopkgbuild"
  13. )
  14. // upgrade type describes a system upgrade.
  15. type upgrade struct {
  16. Name string
  17. Repository string
  18. LocalVersion string
  19. RemoteVersion string
  20. }
  21. // Slice is a slice of Upgrades
  22. type upSlice []upgrade
  23. func (u upSlice) Len() int { return len(u) }
  24. func (u upSlice) Swap(i, j int) { u[i], u[j] = u[j], u[i] }
  25. func (u upSlice) Less(i, j int) bool {
  26. iRunes := []rune(u[i].Repository)
  27. jRunes := []rune(u[j].Repository)
  28. max := len(iRunes)
  29. if max > len(jRunes) {
  30. max = len(jRunes)
  31. }
  32. for idx := 0; idx < max; idx++ {
  33. ir := iRunes[idx]
  34. jr := jRunes[idx]
  35. lir := unicode.ToLower(ir)
  36. ljr := unicode.ToLower(jr)
  37. if lir != ljr {
  38. return lir > ljr
  39. }
  40. // the lowercase runes are the same, so compare the original
  41. if ir != jr {
  42. return ir > jr
  43. }
  44. }
  45. return false
  46. }
  47. // FilterPackages filters packages based on source and type.
  48. func FilterPackages() (local []alpm.Package, remote []alpm.Package,
  49. localNames []string, remoteNames []string, err error) {
  50. localDb, err := AlpmHandle.LocalDb()
  51. if err != nil {
  52. return
  53. }
  54. dbList, err := AlpmHandle.SyncDbs()
  55. if err != nil {
  56. return
  57. }
  58. f := func(k alpm.Package) error {
  59. found := false
  60. // For each DB search for our secret package.
  61. _ = dbList.ForEach(func(d alpm.Db) error {
  62. if found {
  63. return nil
  64. }
  65. _, err := d.PkgByName(k.Name())
  66. if err == nil {
  67. found = true
  68. local = append(local, k)
  69. localNames = append(localNames, k.Name())
  70. }
  71. return nil
  72. })
  73. if !found {
  74. remote = append(remote, k)
  75. remoteNames = append(remoteNames, k.Name())
  76. }
  77. return nil
  78. }
  79. err = localDb.PkgCache().ForEach(f)
  80. return
  81. }
  82. // Print prints the details of the packages to upgrade.
  83. func (u upSlice) Print(start int) {
  84. for k, i := range u {
  85. old, err := pkgb.NewCompleteVersion(i.LocalVersion)
  86. if err != nil {
  87. fmt.Println(i.Name, err)
  88. }
  89. new, err := pkgb.NewCompleteVersion(i.RemoteVersion)
  90. if err != nil {
  91. fmt.Println(i.Name, err)
  92. }
  93. f := func(name string) (color int) {
  94. var hash = 5381
  95. for i := 0; i < len(name); i++ {
  96. hash = int(name[i]) + ((hash << 5) + (hash))
  97. }
  98. return hash%6 + 31
  99. }
  100. fmt.Printf("\x1b[33m%-2d\x1b[0m ", len(u)+start-k-1)
  101. fmt.Printf("\x1b[1;%dm%s\x1b[0m/\x1b[1;39m%-25s\t\t\x1b[0m", f(i.Repository), i.Repository, i.Name)
  102. if old.Version != new.Version {
  103. fmt.Printf("\x1b[31m%18s\x1b[0m-%d -> \x1b[1;32m%s\x1b[0m-%d\x1b[0m",
  104. old.Version, old.Pkgrel,
  105. new.Version, new.Pkgrel)
  106. } else {
  107. fmt.Printf("\x1b[0m%18s-\x1b[31m%d\x1b[0m -> %s-\x1b[32m%d\x1b[0m",
  108. old.Version, old.Pkgrel,
  109. new.Version, new.Pkgrel)
  110. }
  111. print("\n")
  112. }
  113. }
  114. // List returns lists of packages to upgrade from each source.
  115. func upList() (aurUp upSlice, repoUp upSlice, err error) {
  116. local, remote, _, remoteNames, err := FilterPackages()
  117. if err != nil {
  118. return
  119. }
  120. repoC := make(chan upSlice)
  121. aurC := make(chan upSlice)
  122. errC := make(chan error)
  123. fmt.Println("\x1b[1;36;1m::\x1b[0m\x1b[1m Searching databases for updates...\x1b[0m")
  124. go func() {
  125. repoUpList, err := upRepo(local)
  126. errC <- err
  127. repoC <- repoUpList
  128. }()
  129. fmt.Println("\x1b[1;36;1m::\x1b[0m\x1b[1m Searching AUR for updates...\x1b[0m")
  130. go func() {
  131. aurUpList, err := upAUR(remote, remoteNames)
  132. errC <- err
  133. aurC <- aurUpList
  134. }()
  135. var i = 0
  136. loop:
  137. for {
  138. select {
  139. case repoUp = <-repoC:
  140. i++
  141. case aurUp = <-aurC:
  142. i++
  143. case err := <-errC:
  144. if err != nil {
  145. fmt.Println(err)
  146. }
  147. default:
  148. if i == 2 {
  149. close(repoC)
  150. close(aurC)
  151. close(errC)
  152. break loop
  153. }
  154. }
  155. }
  156. return
  157. }
  158. // aur gathers foreign packages and checks if they have new versions.
  159. // Output: Upgrade type package list.
  160. func upAUR(remote []alpm.Package, remoteNames []string) (toUpgrade upSlice, err error) {
  161. var j int
  162. var routines int
  163. var routineDone int
  164. packageC := make(chan upgrade)
  165. done := make(chan bool)
  166. for i := len(remote); i != 0; i = j {
  167. //Split requests so AUR RPC doesn't get mad at us.
  168. j = i - config.RequestSplitN
  169. if j < 0 {
  170. j = 0
  171. }
  172. routines++
  173. go func(local []alpm.Package, remote []string) {
  174. qtemp, err := rpc.Info(remoteNames)
  175. if err != nil {
  176. fmt.Println(err)
  177. done <- true
  178. return
  179. }
  180. // For each item in query: Search equivalent in foreign.
  181. // We assume they're ordered and are returned ordered
  182. // and will only be missing if they don't exist in AUR.
  183. max := len(qtemp) - 1
  184. var missing, x int
  185. for i := range local {
  186. x = i - missing
  187. if x > max {
  188. break
  189. } else if qtemp[x].Name == local[i].Name() {
  190. if (config.TimeUpdate && (int64(qtemp[x].LastModified) > local[i].BuildDate().Unix())) ||
  191. (alpm.VerCmp(local[i].Version(), qtemp[x].Version) < 0) {
  192. packageC <- upgrade{qtemp[x].Name, "aur", local[i].Version(), qtemp[x].Version}
  193. }
  194. continue
  195. } else {
  196. missing++
  197. }
  198. }
  199. done <- true
  200. }(remote[j:i], remoteNames[j:i])
  201. }
  202. for {
  203. select {
  204. case pkg := <-packageC:
  205. toUpgrade = append(toUpgrade, pkg)
  206. case <-done:
  207. routineDone++
  208. if routineDone == routines {
  209. err = nil
  210. return
  211. }
  212. }
  213. }
  214. }
  215. // repo gathers local packages and checks if they have new versions.
  216. // Output: Upgrade type package list.
  217. func upRepo(local []alpm.Package) (upSlice, error) {
  218. dbList, err := AlpmHandle.SyncDbs()
  219. if err != nil {
  220. return nil, err
  221. }
  222. slice := upSlice{}
  223. primeloop:
  224. for _, pkg := range local {
  225. newPkg := pkg.NewVersion(dbList)
  226. if newPkg != nil {
  227. for _, ignorePkg := range AlpmConf.IgnorePkg {
  228. if pkg.Name() == ignorePkg {
  229. fmt.Printf("\x1b[33mwarning:\x1b[0m %s (ignored pkg) ignoring upgrade (%s -> %s)\n", pkg.Name(), pkg.Version(), newPkg.Version())
  230. continue primeloop
  231. }
  232. }
  233. for _, ignoreGroup := range AlpmConf.IgnoreGroup {
  234. for _, group := range pkg.Groups().Slice() {
  235. if group == ignoreGroup {
  236. fmt.Printf("\x1b[33mwarning:\x1b[0m %s (ignored group) ignoring upgrade (%s -> %s)\n", pkg.Name(), pkg.Version(), newPkg.Version())
  237. continue primeloop
  238. }
  239. }
  240. }
  241. slice = append(slice, upgrade{pkg.Name(), newPkg.DB().Name(), pkg.Version(), newPkg.Version()})
  242. }
  243. }
  244. return slice, nil
  245. }
  246. // Upgrade handles updating the cache and installing updates.
  247. func upgradePkgs(flags []string) error {
  248. aurUp, repoUp, err := upList()
  249. if err != nil {
  250. return err
  251. } else if len(aurUp)+len(repoUp) == 0 {
  252. fmt.Println("\nthere is nothing to do")
  253. return err
  254. }
  255. sort.Sort(repoUp)
  256. fmt.Printf("\x1b[1;34;1m:: \x1b[0m\x1b[1m%d Packages to upgrade.\x1b[0m\n", len(aurUp)+len(repoUp))
  257. repoUp.Print(len(aurUp))
  258. aurUp.Print(0)
  259. fmt.Print("\x1b[32mEnter packages you don't want to upgrade.\x1b[0m\nNumbers: ")
  260. reader := bufio.NewReader(os.Stdin)
  261. numberBuf, overflow, err := reader.ReadLine()
  262. if err != nil || overflow {
  263. fmt.Println(err)
  264. return err
  265. }
  266. result := strings.Fields(string(numberBuf))
  267. var repoNums []int
  268. var aurNums []int
  269. for _, numS := range result {
  270. num, err := strconv.Atoi(numS)
  271. if err != nil {
  272. continue
  273. }
  274. if num > len(aurUp)+len(repoUp)-1 || num < 0 {
  275. continue
  276. } else if num < len(aurUp) {
  277. num = len(aurUp) - num - 1
  278. aurNums = append(aurNums, num)
  279. } else {
  280. num = len(aurUp) + len(repoUp) - num - 1
  281. repoNums = append(repoNums, num)
  282. }
  283. }
  284. if len(repoUp) != 0 {
  285. var repoNames []string
  286. repoloop:
  287. for i, k := range repoUp {
  288. for _, j := range repoNums {
  289. if j == i {
  290. continue repoloop
  291. }
  292. }
  293. repoNames = append(repoNames, k.Name)
  294. }
  295. err := passToPacman("-S", repoNames, flags)
  296. if err != nil {
  297. fmt.Println("Error upgrading repo packages.")
  298. }
  299. }
  300. if len(aurUp) != 0 {
  301. var aurNames []string
  302. aurloop:
  303. for i, k := range aurUp {
  304. for _, j := range aurNums {
  305. if j == i {
  306. continue aurloop
  307. }
  308. }
  309. aurNames = append(aurNames, k.Name)
  310. }
  311. aurInstall(aurNames, flags)
  312. }
  313. return nil
  314. }
  315. func develUpgrade(foreign map[string]alpm.Package, flags []string) error {
  316. fmt.Println(" Checking development packages...")
  317. develUpdates := checkUpdates(foreign)
  318. if len(develUpdates) != 0 {
  319. for _, q := range develUpdates {
  320. fmt.Printf("\x1b[1m\x1b[32m==>\x1b[33;1m %s\x1b[0m\n", q)
  321. }
  322. // Install updated packages
  323. if !continueTask("Proceed with upgrade?", "nN") {
  324. return nil
  325. }
  326. err := aurInstall(develUpdates, flags)
  327. if err != nil {
  328. fmt.Println(err)
  329. }
  330. }
  331. return nil
  332. }