cmd.go 17 KB

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