cmd.go 15 KB

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