clean.go 5.6 KB

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