clean.go 5.7 KB

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