actions.go 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357
  1. package yay
  2. import (
  3. "bufio"
  4. "fmt"
  5. "io"
  6. "math"
  7. "os"
  8. "os/exec"
  9. "strconv"
  10. "strings"
  11. "time"
  12. aur "github.com/jguer/yay/aur"
  13. pac "github.com/jguer/yay/pacman"
  14. "github.com/jguer/yay/util"
  15. )
  16. // NumberMenu presents a CLI for selecting packages to install.
  17. func NumberMenu(pkgS []string, flags []string) (err error) {
  18. var num int
  19. aq, numaq, err := aur.Search(pkgS, true)
  20. if err != nil {
  21. fmt.Println("Error during AUR search:", err)
  22. }
  23. pq, numpq, err := pac.Search(pkgS)
  24. if err != nil {
  25. return
  26. }
  27. if numpq == 0 && numaq == 0 {
  28. return fmt.Errorf("no packages match search")
  29. }
  30. if util.SortMode == util.BottomUp {
  31. aq.PrintSearch(numpq)
  32. pq.PrintSearch()
  33. } else {
  34. pq.PrintSearch()
  35. aq.PrintSearch(numpq)
  36. }
  37. fmt.Printf("\x1b[32m%s\x1b[0m\nNumbers:", "Type numbers to install. Separate each number with a space.")
  38. reader := bufio.NewReader(os.Stdin)
  39. numberBuf, overflow, err := reader.ReadLine()
  40. if err != nil || overflow {
  41. fmt.Println(err)
  42. return
  43. }
  44. numberString := string(numberBuf)
  45. var aurInstall []string
  46. var repoInstall []string
  47. result := strings.Fields(numberString)
  48. for _, numS := range result {
  49. num, err = strconv.Atoi(numS)
  50. if err != nil {
  51. continue
  52. }
  53. // Install package
  54. if num > numaq+numpq-1 || num < 0 {
  55. continue
  56. } else if num > numpq-1 {
  57. if util.SortMode == util.BottomUp {
  58. aurInstall = append(aurInstall, aq[numaq+numpq-num-1].Name)
  59. } else {
  60. aurInstall = append(aurInstall, aq[num-numpq].Name)
  61. }
  62. } else {
  63. if util.SortMode == util.BottomUp {
  64. repoInstall = append(repoInstall, pq[numpq-num-1].Name)
  65. } else {
  66. repoInstall = append(repoInstall, pq[num].Name)
  67. }
  68. }
  69. }
  70. if len(repoInstall) != 0 {
  71. pac.Install(repoInstall, flags)
  72. }
  73. if len(aurInstall) != 0 {
  74. q, n, err := aur.MultiInfo(aurInstall)
  75. if err != nil {
  76. return err
  77. } else if n != len(aurInstall) {
  78. q.MissingPackage(aurInstall)
  79. }
  80. var finalrm []string
  81. for _, aurpkg := range q {
  82. finalmdeps, err := aurpkg.Install(flags)
  83. finalrm = append(finalrm, finalmdeps...)
  84. if err != nil {
  85. // Do not abandon program, we might still be able to install the rest
  86. fmt.Println(err)
  87. }
  88. }
  89. if len(finalrm) != 0 {
  90. aur.RemoveMakeDeps(finalrm)
  91. }
  92. }
  93. return nil
  94. }
  95. // Install handles package installs
  96. func Install(pkgs []string, flags []string) error {
  97. aurs, repos, _ := pac.PackageSlices(pkgs)
  98. err := pac.Install(repos, flags)
  99. if err != nil {
  100. fmt.Println("Error installing repo packages.")
  101. }
  102. q, n, err := aur.MultiInfo(aurs)
  103. if len(aurs) != n || err != nil {
  104. fmt.Println("Unable to get info on some packages")
  105. }
  106. var finalrm []string
  107. for _, aurpkg := range q {
  108. finalmdeps, err := aurpkg.Install(flags)
  109. finalrm = append(finalrm, finalmdeps...)
  110. if err != nil {
  111. fmt.Println("Error installing", aurpkg.Name, ":", err)
  112. }
  113. }
  114. if len(finalrm) != 0 {
  115. aur.RemoveMakeDeps(finalrm)
  116. }
  117. return nil
  118. }
  119. // Upgrade handles updating the cache and installing updates.
  120. func Upgrade(flags []string) error {
  121. errp := pac.UpdatePackages(flags)
  122. erra := aur.Upgrade(flags)
  123. if errp != nil {
  124. return errp
  125. }
  126. return erra
  127. }
  128. // SyncSearch presents a query to the local repos and to the AUR.
  129. func SyncSearch(pkgS []string) (err error) {
  130. aq, _, err := aur.Search(pkgS, true)
  131. if err != nil {
  132. return err
  133. }
  134. pq, _, err := pac.Search(pkgS)
  135. if err != nil {
  136. return err
  137. }
  138. if util.SortMode == util.BottomUp {
  139. aq.PrintSearch(0)
  140. pq.PrintSearch()
  141. } else {
  142. pq.PrintSearch()
  143. aq.PrintSearch(0)
  144. }
  145. return nil
  146. }
  147. // SyncInfo serves as a pacman -Si for repo packages and AUR packages.
  148. func SyncInfo(pkgS []string, flags []string) (err error) {
  149. aurS, repoS, err := pac.PackageSlices(pkgS)
  150. if err != nil {
  151. return
  152. }
  153. q, _, err := aur.MultiInfo(aurS)
  154. if err != nil {
  155. fmt.Println(err)
  156. }
  157. for _, aurP := range q {
  158. aurP.PrintInfo()
  159. }
  160. if len(repoS) != 0 {
  161. err = PassToPacman("-Si", repoS, flags)
  162. }
  163. return
  164. }
  165. // LocalStatistics returns installed packages statistics.
  166. func LocalStatistics(version string) error {
  167. info, err := pac.Statistics()
  168. if err != nil {
  169. return err
  170. }
  171. foreignS, foreign, _ := pac.ForeignPackages()
  172. fmt.Printf("\n Yay version r%s\n", version)
  173. fmt.Println("\x1B[1;34m===========================================\x1B[0m")
  174. fmt.Printf("\x1B[1;32mTotal installed packages: \x1B[0;33m%d\x1B[0m\n", info.Totaln)
  175. fmt.Printf("\x1B[1;32mTotal foreign installed packages: \x1B[0;33m%d\x1B[0m\n", foreign)
  176. fmt.Printf("\x1B[1;32mExplicitly installed packages: \x1B[0;33m%d\x1B[0m\n", info.Expln)
  177. fmt.Printf("\x1B[1;32mTotal Size occupied by packages: \x1B[0;33m%s\x1B[0m\n", size(info.TotalSize))
  178. fmt.Println("\x1B[1;34m===========================================\x1B[0m")
  179. fmt.Println("\x1B[1;32mTen biggest packages\x1B[0m")
  180. pac.BiggestPackages()
  181. fmt.Println("\x1B[1;34m===========================================\x1B[0m")
  182. keys := make([]string, len(foreignS))
  183. i := 0
  184. for k := range foreignS {
  185. keys[i] = k
  186. i++
  187. }
  188. q, _, err := aur.MultiInfo(keys)
  189. if err != nil {
  190. return err
  191. }
  192. for _, res := range q {
  193. if res.Maintainer == "" {
  194. fmt.Printf("\x1b[1;31;40mWarning: \x1B[1;33;40m%s\x1b[0;37;40m is orphaned.\x1b[0m\n", res.Name)
  195. }
  196. if res.OutOfDate != 0 {
  197. fmt.Printf("\x1b[1;31;40mWarning: \x1B[1;33;40m%s\x1b[0;37;40m is out-of-date in AUR.\x1b[0m\n", res.Name)
  198. }
  199. }
  200. return nil
  201. }
  202. // Function by pyk https://github.com/pyk/byten
  203. func index(s int64) float64 {
  204. x := math.Log(float64(s)) / math.Log(1024)
  205. return math.Floor(x)
  206. }
  207. // Function by pyk https://github.com/pyk/byten
  208. func countSize(s int64, i float64) float64 {
  209. return float64(s) / math.Pow(1024, math.Floor(i))
  210. }
  211. // Size return a formated string from file size
  212. // Function by pyk https://github.com/pyk/byten
  213. func size(s int64) string {
  214. symbols := []string{"B", "KB", "MB", "GB", "TB", "PB", "EB"}
  215. i := index(s)
  216. if s < 10 {
  217. return fmt.Sprintf("%dB", s)
  218. }
  219. size := countSize(s, i)
  220. format := "%.0f"
  221. if size < 10 {
  222. format = "%.1f"
  223. }
  224. return fmt.Sprintf(format+"%s", size, symbols[int(i)])
  225. }
  226. // PassToPacman outsorces execution to pacman binary without modifications.
  227. func PassToPacman(op string, pkgs []string, flags []string) error {
  228. var cmd *exec.Cmd
  229. var args []string
  230. args = append(args, op)
  231. if len(pkgs) != 0 {
  232. args = append(args, pkgs...)
  233. }
  234. if len(flags) != 0 {
  235. args = append(args, flags...)
  236. }
  237. if strings.Contains(op, "-Q") || op == "-Si" {
  238. cmd = exec.Command("pacman", args...)
  239. } else {
  240. args = append([]string{"pacman"}, args...)
  241. cmd = exec.Command("sudo", args...)
  242. }
  243. cmd.Stdout = os.Stdout
  244. cmd.Stdin = os.Stdin
  245. cmd.Stderr = os.Stderr
  246. err := cmd.Run()
  247. return err
  248. }
  249. // CleanDependencies removels all dangling dependencies in system
  250. func CleanDependencies(pkgs []string) error {
  251. hanging, err := pac.HangingPackages()
  252. if err != nil {
  253. return err
  254. }
  255. if len(hanging) != 0 {
  256. if !util.ContinueTask("Confirm Removal?", "nN") {
  257. return nil
  258. }
  259. err = pac.CleanRemove(hanging)
  260. }
  261. return err
  262. }
  263. // GetPkgbuild gets the pkgbuild of the package 'pkg' trying the ABS first and then the AUR trying the ABS first and then the AUR.
  264. func GetPkgbuild(pkg string) (err error) {
  265. wd, err := os.Getwd()
  266. if err != nil {
  267. return
  268. }
  269. wd = wd + "/"
  270. err = pac.GetPkgbuild(pkg, wd)
  271. if err == nil {
  272. return
  273. }
  274. err = aur.GetPkgbuild(pkg, wd)
  275. return
  276. }
  277. // Complete provides completion info for shells
  278. func Complete() (err error) {
  279. path := os.Getenv("HOME") + "/.cache/yay/aur.cache"
  280. if info, err := os.Stat(path); os.IsNotExist(err) || time.Since(info.ModTime()).Hours() > 48 {
  281. os.MkdirAll(os.Getenv("HOME")+"/.cache/yay", 0755)
  282. out, err := os.Create(path)
  283. if err != nil {
  284. return err
  285. }
  286. defer out.Close()
  287. aur.CreateAURList(out)
  288. err = pac.CreatePackageList(out)
  289. return err
  290. }
  291. in, err := os.OpenFile(path, os.O_RDWR|os.O_CREATE, 0755)
  292. if err != nil {
  293. return err
  294. }
  295. defer in.Close()
  296. _, err = io.Copy(os.Stdout, in)
  297. return err
  298. }