actions.go 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  1. package yay
  2. import (
  3. "bufio"
  4. "fmt"
  5. "math"
  6. "os"
  7. "os/exec"
  8. "strconv"
  9. "strings"
  10. "github.com/jguer/yay/aur"
  11. pac "github.com/jguer/yay/pacman"
  12. )
  13. // BuildDir is the root for package building
  14. const BuildDir string = "/tmp/yaytmp/"
  15. // SearchMode is search without numbers.
  16. const SearchMode int = -1
  17. // SortMode NumberMenu and Search
  18. var SortMode = DownTop
  19. // NoConfirm ignores prompts.
  20. var NoConfirm = false
  21. // Determines NumberMenu and Search Order
  22. const (
  23. DownTop = iota
  24. TopDown
  25. )
  26. // Config copies settings over to AUR and Pacman packages
  27. func Config() {
  28. aur.SortMode = SortMode
  29. pac.SortMode = SortMode
  30. aur.NoConfirm = NoConfirm
  31. pac.NoConfirm = NoConfirm
  32. }
  33. // NumberMenu presents a CLI for selecting packages to install.
  34. func NumberMenu(pkgName string, flags []string) (err error) {
  35. var num int
  36. var numberString string
  37. a, nA, err := aur.Search(pkgName, true)
  38. if err != nil {
  39. fmt.Println("Error during AUR search:", err)
  40. }
  41. r, nR, err := pac.Search(pkgName)
  42. if err != nil {
  43. return
  44. }
  45. if nR == 0 && nA == 0 {
  46. return fmt.Errorf("no packages match search")
  47. }
  48. if aur.SortMode == aur.DownTop {
  49. a.PrintSearch(nR)
  50. r.PrintSearch(0)
  51. } else {
  52. r.PrintSearch(0)
  53. a.PrintSearch(nR)
  54. }
  55. fmt.Printf("\x1b[32m%s\x1b[0m\nNumbers:", "Type numbers to install. Separate each number with a space.")
  56. reader := bufio.NewReader(os.Stdin)
  57. numberString, err = reader.ReadString('\n')
  58. if err != nil {
  59. fmt.Println(err)
  60. return
  61. }
  62. var aurInstall []aur.Result
  63. var repoInstall []string
  64. result := strings.Fields(numberString)
  65. for _, numS := range result {
  66. num, err = strconv.Atoi(numS)
  67. if err != nil {
  68. continue
  69. }
  70. // Install package
  71. if num > nA+nR-1 || num < 0 {
  72. continue
  73. } else if num > nR-1 {
  74. if aur.SortMode == aur.DownTop {
  75. aurInstall = append(aurInstall, a[nA+nR-num-1])
  76. } else {
  77. aurInstall = append(aurInstall, a[num-nR])
  78. }
  79. } else {
  80. if aur.SortMode == aur.DownTop {
  81. repoInstall = append(repoInstall, r[nR-num-1].Name)
  82. } else {
  83. repoInstall = append(repoInstall, r[num].Name)
  84. }
  85. }
  86. }
  87. if len(repoInstall) != 0 {
  88. pac.Install(repoInstall, flags)
  89. }
  90. for _, aurpkg := range aurInstall {
  91. err = aurpkg.Install(BuildDir, flags)
  92. if err != nil {
  93. // Do not abandon program, we might still be able to install the rest
  94. fmt.Println(err)
  95. }
  96. }
  97. return
  98. }
  99. // Install handles package installs
  100. func Install(pkgs []string, flags []string) error {
  101. aurs, repos, _ := pac.PackageSlices(pkgs)
  102. err := pac.Install(repos, flags)
  103. if err != nil {
  104. fmt.Println("Error installing repo packages.")
  105. }
  106. q, n, err := aur.MultiInfo(aurs)
  107. if len(aurs) != n || err != nil {
  108. fmt.Println("Unable to get info on some packages")
  109. }
  110. for _, aurpkg := range q {
  111. err = aurpkg.Install(BuildDir, flags)
  112. if err != nil {
  113. fmt.Println("Error installing", aurpkg.Name, ":", err)
  114. }
  115. }
  116. return nil
  117. }
  118. // Upgrade handles updating the cache and installing updates.
  119. func Upgrade(flags []string) error {
  120. errp := pac.UpdatePackages(flags)
  121. erra := aur.Upgrade(BuildDir, flags)
  122. if errp != nil {
  123. return errp
  124. }
  125. return erra
  126. }
  127. // Search presents a query to the local repos and to the AUR.
  128. func Search(pkg string) (err error) {
  129. a, _, err := aur.Search(pkg, true)
  130. if err != nil {
  131. return err
  132. }
  133. r, _, err := pac.Search(pkg)
  134. if err != nil {
  135. return err
  136. }
  137. if SortMode == aur.DownTop {
  138. a.PrintSearch(SearchMode)
  139. r.PrintSearch(SearchMode)
  140. } else {
  141. r.PrintSearch(SearchMode)
  142. a.PrintSearch(SearchMode)
  143. }
  144. return nil
  145. }
  146. // LocalStatistics returns installed packages statistics.
  147. func LocalStatistics(version string) error {
  148. pkgmap, info, err := pac.Statistics()
  149. if err != nil {
  150. return err
  151. }
  152. _, foreign, _ := pac.ForeignPackages()
  153. fmt.Printf("\n Yay version r%s\n", version)
  154. fmt.Println("\x1B[1;34m===========================================\x1B[0m")
  155. fmt.Printf("\x1B[1;32mTotal installed packages: \x1B[0;33m%d\x1B[0m\n", info.Totaln)
  156. fmt.Printf("\x1B[1;32mTotal foreign installed packages: \x1B[0;33m%d\x1B[0m\n", foreign)
  157. fmt.Printf("\x1B[1;32mExplicitly installed packages: \x1B[0;33m%d\x1B[0m\n", info.Expln)
  158. fmt.Printf("\x1B[1;32mTotal Size occupied by packages: \x1B[0;33m%s\x1B[0m\n", size(info.TotalSize))
  159. fmt.Println("\x1B[1;34m===========================================\x1B[0m")
  160. fmt.Println("\x1B[1;32mTen biggest packages\x1B[0m")
  161. for name, psize := range pkgmap {
  162. fmt.Printf("%s: \x1B[0;33m%s\x1B[0m\n", name, size(psize))
  163. }
  164. fmt.Println("\x1B[1;34m===========================================\x1B[0m")
  165. return nil
  166. }
  167. // Function by pyk https://github.com/pyk/byten
  168. func index(s int64) float64 {
  169. x := math.Log(float64(s)) / math.Log(1024)
  170. return math.Floor(x)
  171. }
  172. // Function by pyk https://github.com/pyk/byten
  173. func countSize(s int64, i float64) float64 {
  174. return float64(s) / math.Pow(1024, math.Floor(i))
  175. }
  176. // Size return a formated string from file size
  177. // Function by pyk https://github.com/pyk/byten
  178. func size(s int64) string {
  179. symbols := []string{"B", "KB", "MB", "GB", "TB", "PB", "EB"}
  180. i := index(s)
  181. if s < 10 {
  182. return fmt.Sprintf("%dB", s)
  183. }
  184. size := countSize(s, i)
  185. format := "%.0f"
  186. if size < 10 {
  187. format = "%.1f"
  188. }
  189. return fmt.Sprintf(format+"%s", size, symbols[int(i)])
  190. }
  191. // PassToPacman outsorces execution to pacman binary without modifications.
  192. func PassToPacman(op string, pkgs []string, flags []string) error {
  193. var cmd *exec.Cmd
  194. var args []string
  195. args = append(args, op)
  196. if len(pkgs) != 0 {
  197. args = append(args, pkgs...)
  198. }
  199. if len(flags) != 0 {
  200. args = append(args, flags...)
  201. }
  202. if strings.Contains(op, "-Q") {
  203. cmd = exec.Command("pacman", args...)
  204. } else {
  205. args = append([]string{"pacman"}, args...)
  206. cmd = exec.Command("sudo", args...)
  207. }
  208. cmd.Stdout = os.Stdout
  209. cmd.Stdin = os.Stdin
  210. cmd.Stderr = os.Stderr
  211. err := cmd.Run()
  212. return err
  213. }