clean.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  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, _ := query.GetRemotePackages(config.Runtime.DBExecutor)
  97. files, err := ioutil.ReadDir(config.BuildDir)
  98. if err != nil {
  99. return err
  100. }
  101. cachedPackages := make([]string, 0, len(files))
  102. for _, file := range files {
  103. if !file.IsDir() {
  104. continue
  105. }
  106. cachedPackages = append(cachedPackages, file.Name())
  107. }
  108. // Most people probably don't use keep current and that is the only
  109. // case where this is needed.
  110. // Querying the AUR is slow and needs internet so don't do it if we
  111. // don't need to.
  112. if keepCurrent {
  113. info, errInfo := query.AURInfo(cachedPackages, &query.AURWarnings{}, config.RequestSplitN)
  114. if errInfo != nil {
  115. return errInfo
  116. }
  117. for _, pkg := range info {
  118. inAURBases.Set(pkg.PackageBase)
  119. }
  120. }
  121. for _, pkg := range remotePackages {
  122. if pkg.Base() != "" {
  123. installedBases.Set(pkg.Base())
  124. } else {
  125. installedBases.Set(pkg.Name())
  126. }
  127. }
  128. for _, file := range files {
  129. if !file.IsDir() {
  130. continue
  131. }
  132. if !removeAll {
  133. if keepInstalled && installedBases.Get(file.Name()) {
  134. continue
  135. }
  136. if keepCurrent && inAURBases.Get(file.Name()) {
  137. continue
  138. }
  139. }
  140. err = os.RemoveAll(filepath.Join(config.BuildDir, file.Name()))
  141. if err != nil {
  142. return nil
  143. }
  144. }
  145. return nil
  146. }
  147. func cleanUntracked() error {
  148. fmt.Println(gotext.Get("removing untracked AUR files from cache..."))
  149. files, err := ioutil.ReadDir(config.BuildDir)
  150. if err != nil {
  151. return err
  152. }
  153. for _, file := range files {
  154. if !file.IsDir() {
  155. continue
  156. }
  157. dir := filepath.Join(config.BuildDir, file.Name())
  158. if isGitRepository(dir) {
  159. if err := show(passToGit(dir, "clean", "-fx")); err != nil {
  160. return err
  161. }
  162. }
  163. }
  164. return nil
  165. }
  166. func isGitRepository(dir string) bool {
  167. _, err := os.Stat(filepath.Join(dir, ".git"))
  168. return !os.IsNotExist(err)
  169. }
  170. func cleanAfter(bases []dep.Base) {
  171. fmt.Println(gotext.Get("removing untracked AUR files from cache..."))
  172. for i, base := range bases {
  173. dir := filepath.Join(config.BuildDir, base.Pkgbase())
  174. if !isGitRepository(dir) {
  175. continue
  176. }
  177. text.OperationInfoln(gotext.Get("Cleaning (%d/%d): %s", i+1, len(bases), cyan(dir)))
  178. _, stderr, err := capture(passToGit(dir, "reset", "--hard", "HEAD"))
  179. if err != nil {
  180. text.Errorln(gotext.Get("error resetting %s: %s", base.String(), stderr))
  181. }
  182. if err := show(passToGit(dir, "clean", "-fx", "--exclude='*.pkg.*'")); err != nil {
  183. fmt.Fprintln(os.Stderr, err)
  184. }
  185. }
  186. }
  187. func cleanBuilds(bases []dep.Base) {
  188. for i, base := range bases {
  189. dir := filepath.Join(config.BuildDir, base.Pkgbase())
  190. text.OperationInfoln(gotext.Get("Deleting (%d/%d): %s", i+1, len(bases), cyan(dir)))
  191. if err := os.RemoveAll(dir); err != nil {
  192. fmt.Fprintln(os.Stderr, err)
  193. }
  194. }
  195. }