cmd.go 16 KB

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