clean.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. package main
  2. import (
  3. "context"
  4. "fmt"
  5. "os"
  6. "path/filepath"
  7. "github.com/leonelquinteros/gotext"
  8. "github.com/Jguer/yay/v12/pkg/db"
  9. "github.com/Jguer/yay/v12/pkg/query"
  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, config *settings.Configuration,
  84. keepInstalled, keepCurrent, removeAll bool, dbExecutor db.Executor,
  85. ) error {
  86. fmt.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(config.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 := query.AURInfo(ctx, config.Runtime.AURClient, cachedPackages, query.NewWarnings(nil), config.RequestSplitN)
  107. if errInfo != nil {
  108. return errInfo
  109. }
  110. for i := range info {
  111. inAURBases.Set(info[i].PackageBase)
  112. }
  113. }
  114. for _, pkg := range remotePackages {
  115. if pkg.Base() != "" {
  116. installedBases.Set(pkg.Base())
  117. } else {
  118. installedBases.Set(pkg.Name())
  119. }
  120. }
  121. for _, file := range files {
  122. if !file.IsDir() {
  123. continue
  124. }
  125. if !removeAll {
  126. if keepInstalled && installedBases.Get(file.Name()) {
  127. continue
  128. }
  129. if keepCurrent && inAURBases.Get(file.Name()) {
  130. continue
  131. }
  132. }
  133. dir := filepath.Join(config.BuildDir, file.Name())
  134. err = os.RemoveAll(dir)
  135. if err != nil {
  136. text.Warnln(gotext.Get("Unable to remove %s: %s", dir, err))
  137. }
  138. }
  139. return nil
  140. }
  141. func cleanUntracked(ctx context.Context, cfg *settings.Configuration) error {
  142. fmt.Println(gotext.Get("removing untracked AUR files from cache..."))
  143. files, err := os.ReadDir(cfg.BuildDir)
  144. if err != nil {
  145. return err
  146. }
  147. for _, file := range files {
  148. if !file.IsDir() {
  149. continue
  150. }
  151. dir := filepath.Join(cfg.BuildDir, file.Name())
  152. if isGitRepository(dir) {
  153. if err := cfg.Runtime.CmdBuilder.Show(cfg.Runtime.CmdBuilder.BuildGitCmd(ctx, dir, "clean", "-fx")); err != nil {
  154. text.Warnln(gotext.Get("Unable to clean:"), dir)
  155. return err
  156. }
  157. }
  158. }
  159. return nil
  160. }
  161. func isGitRepository(dir string) bool {
  162. _, err := os.Stat(filepath.Join(dir, ".git"))
  163. return !os.IsNotExist(err)
  164. }
  165. func cleanAfter(ctx context.Context, config *settings.Configuration,
  166. cmdBuilder exe.ICmdBuilder, pkgbuildDirs map[string]string,
  167. ) {
  168. fmt.Println(gotext.Get("removing untracked AUR files from cache..."))
  169. i := 0
  170. for _, dir := range pkgbuildDirs {
  171. text.OperationInfoln(gotext.Get("Cleaning (%d/%d): %s", i+1, len(pkgbuildDirs), text.Cyan(dir)))
  172. _, stderr, err := cmdBuilder.Capture(
  173. cmdBuilder.BuildGitCmd(
  174. ctx, dir, "reset", "--hard", "HEAD"))
  175. if err != nil {
  176. text.Errorln(gotext.Get("error resetting %s: %s", dir, stderr))
  177. }
  178. if err := config.Runtime.CmdBuilder.Show(
  179. config.Runtime.CmdBuilder.BuildGitCmd(
  180. ctx, dir, "clean", "-fx", "--exclude", "*.pkg.*")); err != nil {
  181. fmt.Fprintln(os.Stderr, err)
  182. }
  183. i++
  184. }
  185. }