actions.go 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  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. // SearchMode is search without numbers.
  14. const SearchMode int = -1
  15. // SortMode NumberMenu and Search
  16. var SortMode = DownTop
  17. // NoConfirm ignores prompts.
  18. var NoConfirm = false
  19. // BaseDir is the default building directory for yay
  20. var BaseDir = "/tmp/yaytmp/"
  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. aur.BaseDir = BaseDir
  33. }
  34. // NumberMenu presents a CLI for selecting packages to install.
  35. func NumberMenu(pkgName string, flags []string) (err error) {
  36. var num int
  37. var numberString string
  38. a, nA, err := aur.Search(pkgName, true)
  39. if err != nil {
  40. fmt.Println("Error during AUR search:", err)
  41. }
  42. r, nR, err := pac.Search(pkgName)
  43. if err != nil {
  44. return
  45. }
  46. if nR == 0 && nA == 0 {
  47. return fmt.Errorf("no packages match search")
  48. }
  49. if aur.SortMode == aur.DownTop {
  50. a.PrintSearch(nR)
  51. r.PrintSearch(0)
  52. } else {
  53. r.PrintSearch(0)
  54. a.PrintSearch(nR)
  55. }
  56. fmt.Printf("\x1b[32m%s\x1b[0m\nNumbers:", "Type numbers to install. Separate each number with a space.")
  57. reader := bufio.NewReader(os.Stdin)
  58. numberString, err = reader.ReadString('\n')
  59. if err != nil {
  60. fmt.Println(err)
  61. return
  62. }
  63. var aurInstall []string
  64. var repoInstall []string
  65. result := strings.Fields(numberString)
  66. for _, numS := range result {
  67. num, err = strconv.Atoi(numS)
  68. if err != nil {
  69. continue
  70. }
  71. // Install package
  72. if num > nA+nR-1 || num < 0 {
  73. continue
  74. } else if num > nR-1 {
  75. if aur.SortMode == aur.DownTop {
  76. aurInstall = append(aurInstall, a[nA+nR-num-1].Name)
  77. } else {
  78. aurInstall = append(aurInstall, a[num-nR].Name)
  79. }
  80. } else {
  81. if aur.SortMode == aur.DownTop {
  82. repoInstall = append(repoInstall, r[nR-num-1].Name)
  83. } else {
  84. repoInstall = append(repoInstall, r[num].Name)
  85. }
  86. }
  87. }
  88. if len(repoInstall) != 0 {
  89. pac.Install(repoInstall, flags)
  90. }
  91. if len(aurInstall) != 0 {
  92. q, n, err := aur.MultiInfo(aurInstall)
  93. if err != nil {
  94. return err
  95. } else if n != len(aurInstall) {
  96. q.MissingPackage(aurInstall)
  97. }
  98. for _, aurpkg := range q {
  99. err = aurpkg.Install(flags)
  100. if err != nil {
  101. // Do not abandon program, we might still be able to install the rest
  102. fmt.Println(err)
  103. }
  104. }
  105. }
  106. return nil
  107. }
  108. // Install handles package installs
  109. func Install(pkgs []string, flags []string) error {
  110. aurs, repos, _ := pac.PackageSlices(pkgs)
  111. err := pac.Install(repos, flags)
  112. if err != nil {
  113. fmt.Println("Error installing repo packages.")
  114. }
  115. q, n, err := aur.MultiInfo(aurs)
  116. if len(aurs) != n || err != nil {
  117. fmt.Println("Unable to get info on some packages")
  118. }
  119. for _, aurpkg := range q {
  120. err = aurpkg.Install(flags)
  121. if err != nil {
  122. fmt.Println("Error installing", aurpkg.Name, ":", err)
  123. }
  124. }
  125. return nil
  126. }
  127. // Upgrade handles updating the cache and installing updates.
  128. func Upgrade(flags []string) error {
  129. errp := pac.UpdatePackages(flags)
  130. erra := aur.Upgrade(flags)
  131. if errp != nil {
  132. return errp
  133. }
  134. return erra
  135. }
  136. // Search presents a query to the local repos and to the AUR.
  137. func Search(pkg string) (err error) {
  138. a, _, err := aur.Search(pkg, true)
  139. if err != nil {
  140. return err
  141. }
  142. r, _, err := pac.Search(pkg)
  143. if err != nil {
  144. return err
  145. }
  146. if SortMode == aur.DownTop {
  147. a.PrintSearch(SearchMode)
  148. r.PrintSearch(SearchMode)
  149. } else {
  150. r.PrintSearch(SearchMode)
  151. a.PrintSearch(SearchMode)
  152. }
  153. return nil
  154. }
  155. // SingleSearch serves as a pacman -Si for repo packages and AUR packages.
  156. func SingleSearch(pkgS []string, flags []string) (err error) {
  157. aurS, repoS, err := pac.PackageSlices(pkgS)
  158. if err != nil {
  159. return
  160. }
  161. q, _, err := aur.MultiInfo(aurS)
  162. if err != nil {
  163. fmt.Println(err)
  164. }
  165. for _, aurP := range q {
  166. aurP.PrintInfo()
  167. }
  168. if len(repoS) != 0 {
  169. err = PassToPacman("-Si", repoS, flags)
  170. }
  171. return
  172. }
  173. // LocalStatistics returns installed packages statistics.
  174. func LocalStatistics(version string) error {
  175. info, err := pac.Statistics()
  176. if err != nil {
  177. return err
  178. }
  179. foreignS, foreign, _ := pac.ForeignPackages()
  180. fmt.Printf("\n Yay version r%s\n", version)
  181. fmt.Println("\x1B[1;34m===========================================\x1B[0m")
  182. fmt.Printf("\x1B[1;32mTotal installed packages: \x1B[0;33m%d\x1B[0m\n", info.Totaln)
  183. fmt.Printf("\x1B[1;32mTotal foreign installed packages: \x1B[0;33m%d\x1B[0m\n", foreign)
  184. fmt.Printf("\x1B[1;32mExplicitly installed packages: \x1B[0;33m%d\x1B[0m\n", info.Expln)
  185. fmt.Printf("\x1B[1;32mTotal Size occupied by packages: \x1B[0;33m%s\x1B[0m\n", size(info.TotalSize))
  186. fmt.Println("\x1B[1;34m===========================================\x1B[0m")
  187. fmt.Println("\x1B[1;32mTen biggest packages\x1B[0m")
  188. pac.BiggestPackages()
  189. fmt.Println("\x1B[1;34m===========================================\x1B[0m")
  190. keys := make([]string, len(foreignS))
  191. i := 0
  192. for k := range foreignS {
  193. keys[i] = k
  194. i++
  195. }
  196. q, _, err := aur.MultiInfo(keys)
  197. if err != nil {
  198. return err
  199. }
  200. for _, res := range q {
  201. if res.Maintainer == "" {
  202. fmt.Printf("\x1b[1;31;40mWarning: \x1B[1;33;40m%s\x1b[0;;40m is orphaned.\x1b[0m\n", res.Name)
  203. }
  204. if res.OutOfDate != 0 {
  205. fmt.Printf("\x1b[1;31;40mWarning: \x1B[1;33;40m%s\x1b[0;;40m is out-of-date in AUR.\x1b[0m\n", res.Name)
  206. }
  207. }
  208. return nil
  209. }
  210. // Function by pyk https://github.com/pyk/byten
  211. func index(s int64) float64 {
  212. x := math.Log(float64(s)) / math.Log(1024)
  213. return math.Floor(x)
  214. }
  215. // Function by pyk https://github.com/pyk/byten
  216. func countSize(s int64, i float64) float64 {
  217. return float64(s) / math.Pow(1024, math.Floor(i))
  218. }
  219. // Size return a formated string from file size
  220. // Function by pyk https://github.com/pyk/byten
  221. func size(s int64) string {
  222. symbols := []string{"B", "KB", "MB", "GB", "TB", "PB", "EB"}
  223. i := index(s)
  224. if s < 10 {
  225. return fmt.Sprintf("%dB", s)
  226. }
  227. size := countSize(s, i)
  228. format := "%.0f"
  229. if size < 10 {
  230. format = "%.1f"
  231. }
  232. return fmt.Sprintf(format+"%s", size, symbols[int(i)])
  233. }
  234. // PassToPacman outsorces execution to pacman binary without modifications.
  235. func PassToPacman(op string, pkgs []string, flags []string) error {
  236. var cmd *exec.Cmd
  237. var args []string
  238. args = append(args, op)
  239. if len(pkgs) != 0 {
  240. args = append(args, pkgs...)
  241. }
  242. if len(flags) != 0 {
  243. args = append(args, flags...)
  244. }
  245. if strings.Contains(op, "-Q") || op == "-Si" {
  246. cmd = exec.Command("pacman", args...)
  247. } else {
  248. args = append([]string{"pacman"}, args...)
  249. cmd = exec.Command("sudo", args...)
  250. }
  251. cmd.Stdout = os.Stdout
  252. cmd.Stdin = os.Stdin
  253. cmd.Stderr = os.Stderr
  254. err := cmd.Run()
  255. return err
  256. }