clean.go 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. package main
  2. import (
  3. "fmt"
  4. "io/ioutil"
  5. "os"
  6. "path/filepath"
  7. "github.com/leonelquinteros/gotext"
  8. "github.com/Jguer/yay/v10/pkg/stringset"
  9. "github.com/Jguer/yay/v10/pkg/text"
  10. )
  11. // GetPkgbuild gets the pkgbuild of the package 'pkg' trying the ABS first and then the AUR trying the ABS first and then the AUR.
  12. // RemovePackage removes package from VCS information
  13. func removeVCSPackage(pkgs []string) {
  14. updated := false
  15. for _, pkgName := range pkgs {
  16. if _, ok := savedInfo[pkgName]; ok {
  17. delete(savedInfo, pkgName)
  18. updated = true
  19. }
  20. }
  21. if updated {
  22. err := saveVCSInfo()
  23. if err != nil {
  24. fmt.Fprintln(os.Stderr, err)
  25. }
  26. }
  27. }
  28. // CleanDependencies removes all dangling dependencies in system
  29. func cleanDependencies(removeOptional bool) error {
  30. hanging, err := hangingPackages(removeOptional)
  31. if err != nil {
  32. return err
  33. }
  34. if len(hanging) != 0 {
  35. return cleanRemove(hanging)
  36. }
  37. return nil
  38. }
  39. // CleanRemove sends a full removal command to pacman with the pkgName slice
  40. func cleanRemove(pkgNames []string) error {
  41. if len(pkgNames) == 0 {
  42. return nil
  43. }
  44. arguments := cmdArgs.copyGlobal()
  45. _ = arguments.addArg("R")
  46. arguments.addTarget(pkgNames...)
  47. return show(passToPacman(arguments))
  48. }
  49. func syncClean(parser *arguments) error {
  50. keepInstalled := false
  51. keepCurrent := false
  52. _, removeAll, _ := parser.getArg("c", "clean")
  53. for _, v := range pacmanConf.CleanMethod {
  54. if v == "KeepInstalled" {
  55. keepInstalled = true
  56. } else if v == "KeepCurrent" {
  57. keepCurrent = true
  58. }
  59. }
  60. if mode == modeRepo || mode == modeAny {
  61. if err := show(passToPacman(parser)); err != nil {
  62. return err
  63. }
  64. }
  65. if !(mode == modeAUR || mode == modeAny) {
  66. return nil
  67. }
  68. var question string
  69. if removeAll {
  70. question = gotext.Get("Do you want to remove ALL AUR packages from cache?")
  71. } else {
  72. question = gotext.Get("Do you want to remove all other AUR packages from cache?")
  73. }
  74. fmt.Println(gotext.Get("\nBuild directory:"), config.BuildDir)
  75. if continueTask(question, true) {
  76. if err := cleanAUR(keepInstalled, keepCurrent, removeAll); err != nil {
  77. return err
  78. }
  79. }
  80. if removeAll {
  81. return nil
  82. }
  83. if continueTask(gotext.Get("Do you want to remove ALL untracked AUR files?"), true) {
  84. return cleanUntracked()
  85. }
  86. return nil
  87. }
  88. func cleanAUR(keepInstalled, keepCurrent, removeAll bool) error {
  89. fmt.Println(gotext.Get("removing AUR packages from cache..."))
  90. installedBases := make(stringset.StringSet)
  91. inAURBases := make(stringset.StringSet)
  92. _, remotePackages, _, _, err := filterPackages()
  93. if err != nil {
  94. return err
  95. }
  96. files, err := ioutil.ReadDir(config.BuildDir)
  97. if err != nil {
  98. return err
  99. }
  100. cachedPackages := make([]string, 0, len(files))
  101. for _, file := range files {
  102. if !file.IsDir() {
  103. continue
  104. }
  105. cachedPackages = append(cachedPackages, file.Name())
  106. }
  107. // Most people probably don't use keep current and that is the only
  108. // case where this is needed.
  109. // Querying the AUR is slow and needs internet so don't do it if we
  110. // don't need to.
  111. if keepCurrent {
  112. info, errInfo := aurInfo(cachedPackages, &aurWarnings{})
  113. if errInfo != nil {
  114. return errInfo
  115. }
  116. for _, pkg := range info {
  117. inAURBases.Set(pkg.PackageBase)
  118. }
  119. }
  120. for _, pkg := range remotePackages {
  121. if pkg.Base() != "" {
  122. installedBases.Set(pkg.Base())
  123. } else {
  124. installedBases.Set(pkg.Name())
  125. }
  126. }
  127. for _, file := range files {
  128. if !file.IsDir() {
  129. continue
  130. }
  131. if !removeAll {
  132. if keepInstalled && installedBases.Get(file.Name()) {
  133. continue
  134. }
  135. if keepCurrent && inAURBases.Get(file.Name()) {
  136. continue
  137. }
  138. }
  139. err = os.RemoveAll(filepath.Join(config.BuildDir, file.Name()))
  140. if err != nil {
  141. return nil
  142. }
  143. }
  144. return nil
  145. }
  146. func cleanUntracked() error {
  147. fmt.Println(gotext.Get("removing untracked AUR files from cache..."))
  148. files, err := ioutil.ReadDir(config.BuildDir)
  149. if err != nil {
  150. return err
  151. }
  152. for _, file := range files {
  153. if !file.IsDir() {
  154. continue
  155. }
  156. dir := filepath.Join(config.BuildDir, file.Name())
  157. if err := show(passToGit(dir, "clean", "-fx")); err != nil {
  158. return err
  159. }
  160. }
  161. return nil
  162. }
  163. func cleanAfter(bases []Base) {
  164. fmt.Println(gotext.Get("removing untracked AUR files from cache..."))
  165. for i, base := range bases {
  166. dir := filepath.Join(config.BuildDir, base.Pkgbase())
  167. text.OperationInfoln(gotext.Get("Cleaning (%d/%d): %s", i+1, len(bases), cyan(dir)))
  168. _, stderr, err := capture(passToGit(dir, "reset", "--hard", "HEAD"))
  169. if err != nil {
  170. text.Errorln(gotext.Get("error resetting %s: %s", base.String(), stderr))
  171. }
  172. if err := show(passToGit(dir, "clean", "-fx")); err != nil {
  173. fmt.Fprintln(os.Stderr, err)
  174. }
  175. }
  176. }
  177. func cleanBuilds(bases []Base) {
  178. for i, base := range bases {
  179. dir := filepath.Join(config.BuildDir, base.Pkgbase())
  180. text.OperationInfoln(gotext.Get("Deleting (%d/%d): %s", i+1, len(bases), cyan(dir)))
  181. if err := os.RemoveAll(dir); err != nil {
  182. fmt.Fprintln(os.Stderr, err)
  183. }
  184. }
  185. }