cmd.go 16 KB

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