clean.go 4.9 KB

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