clean.go 5.5 KB

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