cmd.go 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443
  1. package main
  2. import (
  3. "bufio"
  4. "context"
  5. "fmt"
  6. "net/http"
  7. "strings"
  8. alpm "github.com/Jguer/go-alpm/v2"
  9. "github.com/leonelquinteros/gotext"
  10. "github.com/Jguer/yay/v11/pkg/completion"
  11. "github.com/Jguer/yay/v11/pkg/db"
  12. "github.com/Jguer/yay/v11/pkg/download"
  13. "github.com/Jguer/yay/v11/pkg/intrange"
  14. "github.com/Jguer/yay/v11/pkg/news"
  15. "github.com/Jguer/yay/v11/pkg/query"
  16. "github.com/Jguer/yay/v11/pkg/settings"
  17. "github.com/Jguer/yay/v11/pkg/settings/parser"
  18. "github.com/Jguer/yay/v11/pkg/text"
  19. "github.com/Jguer/yay/v11/pkg/upgrade"
  20. "github.com/Jguer/yay/v11/pkg/vcs"
  21. )
  22. func usage() {
  23. fmt.Println(`Usage:
  24. yay
  25. yay <operation> [...]
  26. yay <package(s)>
  27. operations:
  28. yay {-h --help}
  29. yay {-V --version}
  30. yay {-D --database} <options> <package(s)>
  31. yay {-F --files} [options] [package(s)]
  32. yay {-Q --query} [options] [package(s)]
  33. yay {-R --remove} [options] <package(s)>
  34. yay {-S --sync} [options] [package(s)]
  35. yay {-T --deptest} [options] [package(s)]
  36. yay {-U --upgrade} [options] <file(s)>
  37. New operations:
  38. yay {-Y --yay} [options] [package(s)]
  39. yay {-P --show} [options]
  40. yay {-G --getpkgbuild} [options] [package(s)]
  41. If no arguments are provided 'yay -Syu' will be performed
  42. If no operation is provided -Y will be assumed
  43. New options:
  44. --repo Assume targets are from the repositories
  45. -a --aur Assume targets are from the AUR
  46. Permanent configuration options:
  47. --save Causes the following options to be saved back to the
  48. config file when used
  49. --aururl <url> Set an alternative AUR URL
  50. --builddir <dir> Directory used to download and run PKGBUILDS
  51. --editor <file> Editor to use when editing PKGBUILDs
  52. --editorflags <flags> Pass arguments to editor
  53. --makepkg <file> makepkg command to use
  54. --mflags <flags> Pass arguments to makepkg
  55. --pacman <file> pacman command to use
  56. --git <file> git command to use
  57. --gitflags <flags> Pass arguments to git
  58. --gpg <file> gpg command to use
  59. --gpgflags <flags> Pass arguments to gpg
  60. --config <file> pacman.conf file to use
  61. --makepkgconf <file> makepkg.conf file to use
  62. --nomakepkgconf Use the default makepkg.conf
  63. --requestsplitn <n> Max amount of packages to query per AUR request
  64. --completioninterval <n> Time in days to refresh completion cache
  65. --sortby <field> Sort AUR results by a specific field during search
  66. --searchby <field> Search for packages using a specified field
  67. --answerclean <a> Set a predetermined answer for the clean build menu
  68. --answerdiff <a> Set a predetermined answer for the diff menu
  69. --answeredit <a> Set a predetermined answer for the edit pkgbuild menu
  70. --answerupgrade <a> Set a predetermined answer for the upgrade menu
  71. --noanswerclean Unset the answer for the clean build menu
  72. --noanswerdiff Unset the answer for the edit diff menu
  73. --noansweredit Unset the answer for the edit pkgbuild menu
  74. --noanswerupgrade Unset the answer for the upgrade menu
  75. --cleanmenu Give the option to clean build PKGBUILDS
  76. --diffmenu Give the option to show diffs for build files
  77. --editmenu Give the option to edit/view PKGBUILDS
  78. --upgrademenu Show a detailed list of updates with the option to skip any
  79. --nocleanmenu Don't clean build PKGBUILDS
  80. --nodiffmenu Don't show diffs for build files
  81. --noeditmenu Don't edit/view PKGBUILDS
  82. --noupgrademenu Don't show the upgrade menu
  83. --askremovemake Ask to remove makedepends after install
  84. --removemake Remove makedepends after install
  85. --noremovemake Don't remove makedepends after install
  86. --cleanafter Remove package sources after successful install
  87. --nocleanafter Do not remove package sources after successful build
  88. --bottomup Shows AUR's packages first and then repository's
  89. --topdown Shows repository's packages first and then AUR's
  90. --devel Check development packages during sysupgrade
  91. --nodevel Do not check development packages
  92. --rebuild Always build target packages
  93. --rebuildall Always build all AUR packages
  94. --norebuild Skip package build if in cache and up to date
  95. --rebuildtree Always build all AUR packages even if installed
  96. --redownload Always download pkgbuilds of targets
  97. --noredownload Skip pkgbuild download if in cache and up to date
  98. --redownloadall Always download pkgbuilds of all AUR packages
  99. --provides Look for matching providers when searching for packages
  100. --noprovides Just look for packages by pkgname
  101. --pgpfetch Prompt to import PGP keys from PKGBUILDs
  102. --nopgpfetch Don't prompt to import PGP keys
  103. --useask Automatically resolve conflicts using pacman's ask flag
  104. --nouseask Confirm conflicts manually during the install
  105. --combinedupgrade Refresh then perform the repo and AUR upgrade together
  106. --nocombinedupgrade Perform the repo upgrade and AUR upgrade separately
  107. --batchinstall Build multiple AUR packages then install them together
  108. --nobatchinstall Build and install each AUR package one by one
  109. --sudo <file> sudo command to use
  110. --sudoflags <flags> Pass arguments to sudo
  111. --sudoloop Loop sudo calls in the background to avoid timeout
  112. --nosudoloop Do not loop sudo calls in the background
  113. --timeupdate Check packages' AUR page for changes during sysupgrade
  114. --notimeupdate Do not check packages' AUR page for changes
  115. show specific options:
  116. -c --complete Used for completions
  117. -d --defaultconfig Print default yay configuration
  118. -g --currentconfig Print current yay configuration
  119. -s --stats Display system package statistics
  120. -w --news Print arch news
  121. yay specific options:
  122. -c --clean Remove unneeded dependencies
  123. --gendb Generates development package DB used for updating
  124. getpkgbuild specific options:
  125. -f --force Force download for existing ABS packages
  126. -p --print Print pkgbuild of packages`)
  127. }
  128. func handleCmd(ctx context.Context, cmdArgs *parser.Arguments, dbExecutor db.Executor) error {
  129. if cmdArgs.ExistsArg("h", "help") {
  130. return handleHelp(ctx, cmdArgs)
  131. }
  132. if config.SudoLoop && cmdArgs.NeedRoot(config.Runtime.Mode) {
  133. config.Runtime.CmdBuilder.SudoLoop()
  134. }
  135. switch cmdArgs.Op {
  136. case "V", "version":
  137. handleVersion()
  138. return nil
  139. case "D", "database":
  140. return config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  141. cmdArgs, config.Runtime.Mode, settings.NoConfirm))
  142. case "F", "files":
  143. return config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  144. cmdArgs, config.Runtime.Mode, settings.NoConfirm))
  145. case "Q", "query":
  146. return handleQuery(ctx, cmdArgs, dbExecutor)
  147. case "R", "remove":
  148. return handleRemove(ctx, cmdArgs, config.Runtime.VCSStore)
  149. case "S", "sync":
  150. return handleSync(ctx, cmdArgs, dbExecutor)
  151. case "T", "deptest":
  152. return config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  153. cmdArgs, config.Runtime.Mode, settings.NoConfirm))
  154. case "U", "upgrade":
  155. return config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  156. cmdArgs, config.Runtime.Mode, settings.NoConfirm))
  157. case "G", "getpkgbuild":
  158. return handleGetpkgbuild(ctx, cmdArgs, dbExecutor)
  159. case "P", "show":
  160. return handlePrint(ctx, cmdArgs, dbExecutor)
  161. case "Y", "--yay":
  162. return handleYay(ctx, cmdArgs, dbExecutor)
  163. }
  164. return fmt.Errorf(gotext.Get("unhandled operation"))
  165. }
  166. // getFilter returns filter function which can keep packages which were only
  167. // explicitly installed or ones installed as dependencies for showing available
  168. // updates or their count.
  169. func getFilter(cmdArgs *parser.Arguments) (upgrade.Filter, error) {
  170. deps, explicit := cmdArgs.ExistsArg("d", "deps"), cmdArgs.ExistsArg("e", "explicit")
  171. switch {
  172. case deps && explicit:
  173. return nil, fmt.Errorf(gotext.Get("invalid option: '--deps' and '--explicit' may not be used together"))
  174. case deps:
  175. return func(pkg upgrade.Upgrade) bool {
  176. return pkg.Reason == alpm.PkgReasonDepend
  177. }, nil
  178. case explicit:
  179. return func(pkg upgrade.Upgrade) bool {
  180. return pkg.Reason == alpm.PkgReasonExplicit
  181. }, nil
  182. }
  183. return func(pkg upgrade.Upgrade) bool {
  184. return true
  185. }, nil
  186. }
  187. func handleQuery(ctx context.Context, cmdArgs *parser.Arguments, dbExecutor db.Executor) error {
  188. if cmdArgs.ExistsArg("u", "upgrades") {
  189. filter, err := getFilter(cmdArgs)
  190. if err != nil {
  191. return err
  192. }
  193. return printUpdateList(ctx, cmdArgs, dbExecutor, cmdArgs.ExistsDouble("u", "sysupgrade"), filter)
  194. }
  195. if err := config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  196. cmdArgs, config.Runtime.Mode, settings.NoConfirm)); err != nil {
  197. if str := err.Error(); strings.Contains(str, "exit status") {
  198. // yay -Qdt should not output anything in case of error
  199. return fmt.Errorf("")
  200. }
  201. return err
  202. }
  203. return nil
  204. }
  205. func handleHelp(ctx context.Context, cmdArgs *parser.Arguments) error {
  206. switch cmdArgs.Op {
  207. case "Y", "yay", "G", "getpkgbuild", "P", "show":
  208. usage()
  209. return nil
  210. }
  211. return config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  212. cmdArgs, config.Runtime.Mode, settings.NoConfirm))
  213. }
  214. func handleVersion() {
  215. fmt.Printf("yay v%s - libalpm v%s\n", yayVersion, alpm.Version())
  216. }
  217. func handlePrint(ctx context.Context, cmdArgs *parser.Arguments, dbExecutor db.Executor) error {
  218. switch {
  219. case cmdArgs.ExistsArg("d", "defaultconfig"):
  220. tmpConfig := settings.DefaultConfig()
  221. fmt.Printf("%v", tmpConfig)
  222. return nil
  223. case cmdArgs.ExistsArg("g", "currentconfig"):
  224. fmt.Printf("%v", config)
  225. return nil
  226. case cmdArgs.ExistsArg("n", "numberupgrades"):
  227. filter, err := getFilter(cmdArgs)
  228. if err != nil {
  229. return err
  230. }
  231. return printNumberOfUpdates(ctx, dbExecutor, cmdArgs.ExistsDouble("u", "sysupgrade"), filter)
  232. case cmdArgs.ExistsArg("w", "news"):
  233. double := cmdArgs.ExistsDouble("w", "news")
  234. quiet := cmdArgs.ExistsArg("q", "quiet")
  235. return news.PrintNewsFeed(ctx, config.Runtime.HTTPClient, dbExecutor.LastBuildTime(), config.SortMode, double, quiet)
  236. case cmdArgs.ExistsDouble("c", "complete"):
  237. return completion.Show(ctx, config.Runtime.HTTPClient, dbExecutor,
  238. config.AURURL, config.Runtime.CompletionPath, config.CompletionInterval, true)
  239. case cmdArgs.ExistsArg("c", "complete"):
  240. return completion.Show(ctx, config.Runtime.HTTPClient, dbExecutor,
  241. config.AURURL, config.Runtime.CompletionPath, config.CompletionInterval, false)
  242. case cmdArgs.ExistsArg("s", "stats"):
  243. return localStatistics(ctx, dbExecutor)
  244. }
  245. return nil
  246. }
  247. func handleYay(ctx context.Context, cmdArgs *parser.Arguments, dbExecutor db.Executor) error {
  248. switch {
  249. case cmdArgs.ExistsArg("gendb"):
  250. return createDevelDB(ctx, config, dbExecutor)
  251. case cmdArgs.ExistsDouble("c"):
  252. return cleanDependencies(ctx, cmdArgs, dbExecutor, true)
  253. case cmdArgs.ExistsArg("c", "clean"):
  254. return cleanDependencies(ctx, cmdArgs, dbExecutor, false)
  255. case len(cmdArgs.Targets) > 0:
  256. return handleYogurt(ctx, cmdArgs, dbExecutor)
  257. }
  258. return nil
  259. }
  260. func handleGetpkgbuild(ctx context.Context, cmdArgs *parser.Arguments, dbExecutor download.DBSearcher) error {
  261. if cmdArgs.ExistsArg("p", "print") {
  262. return printPkgbuilds(dbExecutor, config.Runtime.HTTPClient, cmdArgs.Targets, config.Runtime.Mode, config.AURURL)
  263. }
  264. return getPkgbuilds(ctx, dbExecutor, config, cmdArgs.Targets, cmdArgs.ExistsArg("f", "force"))
  265. }
  266. func handleYogurt(ctx context.Context, cmdArgs *parser.Arguments, dbExecutor db.Executor) error {
  267. return displayNumberMenu(ctx, cmdArgs.Targets, dbExecutor, cmdArgs)
  268. }
  269. func handleSync(ctx context.Context, cmdArgs *parser.Arguments, dbExecutor db.Executor) error {
  270. targets := cmdArgs.Targets
  271. switch {
  272. case cmdArgs.ExistsArg("s", "search"):
  273. return syncSearch(ctx, targets, config.Runtime.AURClient, dbExecutor, !cmdArgs.ExistsArg("q", "quiet"))
  274. case cmdArgs.ExistsArg("p", "print", "print-format"):
  275. return config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  276. cmdArgs, config.Runtime.Mode, settings.NoConfirm))
  277. case cmdArgs.ExistsArg("c", "clean"):
  278. return syncClean(ctx, cmdArgs, dbExecutor)
  279. case cmdArgs.ExistsArg("l", "list"):
  280. return syncList(ctx, config.Runtime.HTTPClient, cmdArgs, dbExecutor)
  281. case cmdArgs.ExistsArg("g", "groups"):
  282. return config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  283. cmdArgs, config.Runtime.Mode, settings.NoConfirm))
  284. case cmdArgs.ExistsArg("i", "info"):
  285. return syncInfo(ctx, cmdArgs, targets, dbExecutor)
  286. case cmdArgs.ExistsArg("u", "sysupgrade"):
  287. return install(ctx, cmdArgs, dbExecutor, false)
  288. case len(cmdArgs.Targets) > 0:
  289. return install(ctx, cmdArgs, dbExecutor, false)
  290. case cmdArgs.ExistsArg("y", "refresh"):
  291. return config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  292. cmdArgs, config.Runtime.Mode, settings.NoConfirm))
  293. }
  294. return nil
  295. }
  296. func handleRemove(ctx context.Context, cmdArgs *parser.Arguments, localCache *vcs.InfoStore) error {
  297. err := config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  298. cmdArgs, config.Runtime.Mode, settings.NoConfirm))
  299. if err == nil {
  300. localCache.RemovePackage(cmdArgs.Targets)
  301. }
  302. return err
  303. }
  304. // NumberMenu presents a CLI for selecting packages to install.
  305. func displayNumberMenu(ctx context.Context, pkgS []string, dbExecutor db.Executor, cmdArgs *parser.Arguments) error {
  306. queryBuilder := query.NewSourceQueryBuilder(config.SortMode, config.SortBy, config.Runtime.Mode, config.SearchBy)
  307. queryBuilder.Execute(ctx, dbExecutor, config.Runtime.AURClient, pkgS)
  308. if err := queryBuilder.Results(dbExecutor, query.NumberMenu); err != nil {
  309. return err
  310. }
  311. if queryBuilder.Len() == 0 {
  312. // no results were found
  313. return nil
  314. }
  315. text.Infoln(gotext.Get("Packages to install (eg: 1 2 3, 1-3 or ^4)"))
  316. numberBuf, err := text.GetInput("", false)
  317. if err != nil {
  318. return err
  319. }
  320. include, exclude, _, otherExclude := intrange.ParseNumberMenu(numberBuf)
  321. targets, err := queryBuilder.GetTargets(include, exclude, otherExclude)
  322. if err != nil {
  323. return err
  324. }
  325. arguments := cmdArgs.CopyGlobal()
  326. arguments.AddTarget(targets...)
  327. if len(arguments.Targets) == 0 {
  328. fmt.Println(gotext.Get(" there is nothing to do"))
  329. return nil
  330. }
  331. return install(ctx, arguments, dbExecutor, true)
  332. }
  333. func syncList(ctx context.Context, httpClient *http.Client, cmdArgs *parser.Arguments, dbExecutor db.Executor) error {
  334. aur := false
  335. for i := len(cmdArgs.Targets) - 1; i >= 0; i-- {
  336. if cmdArgs.Targets[i] == "aur" && config.Runtime.Mode.AtLeastAUR() {
  337. cmdArgs.Targets = append(cmdArgs.Targets[:i], cmdArgs.Targets[i+1:]...)
  338. aur = true
  339. }
  340. }
  341. if config.Runtime.Mode.AtLeastAUR() && (len(cmdArgs.Targets) == 0 || aur) {
  342. req, err := http.NewRequestWithContext(ctx, "GET", config.AURURL+"/packages.gz", nil)
  343. if err != nil {
  344. return err
  345. }
  346. resp, err := httpClient.Do(req)
  347. if err != nil {
  348. return err
  349. }
  350. defer resp.Body.Close()
  351. scanner := bufio.NewScanner(resp.Body)
  352. scanner.Scan()
  353. for scanner.Scan() {
  354. name := scanner.Text()
  355. if cmdArgs.ExistsArg("q", "quiet") {
  356. fmt.Println(name)
  357. } else {
  358. fmt.Printf("%s %s %s", text.Magenta("aur"), text.Bold(name), text.Bold(text.Green(gotext.Get("unknown-version"))))
  359. if dbExecutor.LocalPackage(name) != nil {
  360. fmt.Print(text.Bold(text.Blue(gotext.Get(" [Installed]"))))
  361. }
  362. fmt.Println()
  363. }
  364. }
  365. }
  366. if config.Runtime.Mode.AtLeastRepo() && (len(cmdArgs.Targets) != 0 || !aur) {
  367. return config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  368. cmdArgs, config.Runtime.Mode, settings.NoConfirm))
  369. }
  370. return nil
  371. }