cmd.go 17 KB

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