cmd.go 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408
  1. package main
  2. import (
  3. "bufio"
  4. "encoding/json"
  5. "fmt"
  6. "io"
  7. "os"
  8. "path/filepath"
  9. "strconv"
  10. "strings"
  11. "time"
  12. )
  13. func usage() {
  14. fmt.Println(`usage: yay <operation> [...]
  15. operations:
  16. yay {-h --help}
  17. yay {-V --version}
  18. yay {-D --database} <options> <package(s)>
  19. yay {-F --files} [options] [package(s)]
  20. yay {-Q --query} [options] [package(s)]
  21. yay {-R --remove} [options] <package(s)>
  22. yay {-S --sync} [options] [package(s)]
  23. yay {-T --deptest} [options] [package(s)]
  24. yay {-U --upgrade} [options] <file(s)>
  25. New operations:
  26. yay -Qstats displays system information
  27. yay -Cd remove unneeded dependencies
  28. yay -G [package(s)] get pkgbuild from ABS or AUR
  29. yay --gendb generates development package DB used for updating.
  30. Permanent configuration options:
  31. --topdown shows repository's packages first and then aur's
  32. --bottomup shows aur's packages first and then repository's
  33. --devel Check -git/-svn/-hg development version
  34. --nodevel Disable development version checking
  35. --afterclean Clean package sources after successful build
  36. --noafterclean Disable package sources cleaning after successful build
  37. --timeupdate Check package's modification date and version
  38. --notimeupdate Check only package version change
  39. New options:
  40. --noconfirm skip user input on package install
  41. --printconfig Prints current yay configuration
  42. `)
  43. }
  44. func init() {
  45. var configHome string // configHome handles config directory home
  46. var cacheHome string // cacheHome handles cache home
  47. var err error
  48. if 0 == os.Geteuid() {
  49. fmt.Println("Please avoid running yay as root/sudo.")
  50. }
  51. if configHome = os.Getenv("XDG_CONFIG_HOME"); configHome != "" {
  52. if info, err := os.Stat(configHome); err == nil && info.IsDir() {
  53. configHome = configHome + "/yay"
  54. } else {
  55. configHome = os.Getenv("HOME") + "/.config/yay"
  56. }
  57. } else {
  58. configHome = os.Getenv("HOME") + "/.config/yay"
  59. }
  60. if cacheHome = os.Getenv("XDG_CACHE_HOME"); cacheHome != "" {
  61. if info, err := os.Stat(cacheHome); err == nil && info.IsDir() {
  62. cacheHome = cacheHome + "/yay"
  63. } else {
  64. cacheHome = os.Getenv("HOME") + "/.cache/yay"
  65. }
  66. } else {
  67. cacheHome = os.Getenv("HOME") + "/.cache/yay"
  68. }
  69. configFile = configHome + "/config.json"
  70. vcsFile = configHome + "/yay_vcs.json"
  71. completionFile = cacheHome + "/aur_"
  72. ////////////////
  73. // yay config //
  74. ////////////////
  75. defaultSettings(&config)
  76. if _, err = os.Stat(configFile); os.IsNotExist(err) {
  77. err = os.MkdirAll(filepath.Dir(configFile), 0755)
  78. if err != nil {
  79. fmt.Println("Unable to create config directory:", filepath.Dir(configFile), err)
  80. os.Exit(2)
  81. }
  82. // Save the default config if nothing is found
  83. config.saveConfig()
  84. } else {
  85. cfile, errf := os.OpenFile(configFile, os.O_RDWR|os.O_CREATE, 0644)
  86. if errf != nil {
  87. fmt.Println("Error reading config:", err)
  88. } else {
  89. defer cfile.Close()
  90. decoder := json.NewDecoder(cfile)
  91. err = decoder.Decode(&config)
  92. if err != nil {
  93. fmt.Println("Loading default Settings.\nError reading config:", err)
  94. defaultSettings(&config)
  95. }
  96. }
  97. }
  98. /////////////////
  99. // vcs config //
  100. ////////////////
  101. updated = false
  102. vfile, err := os.Open(vcsFile)
  103. if err == nil {
  104. defer vfile.Close()
  105. decoder := json.NewDecoder(vfile)
  106. _ = decoder.Decode(&savedInfo)
  107. }
  108. /////////////////
  109. // alpm config //
  110. /////////////////
  111. alpmConf, err = readAlpmConfig(config.PacmanConf)
  112. if err != nil {
  113. fmt.Println("Unable to read Pacman conf", err)
  114. os.Exit(1)
  115. }
  116. alpmHandle, err = alpmConf.CreateHandle()
  117. if err != nil {
  118. fmt.Println("Unable to CreateHandle", err)
  119. os.Exit(1)
  120. }
  121. }
  122. func parser() (op string, options []string, packages []string, changedConfig bool, err error) {
  123. if len(os.Args) < 2 {
  124. err = fmt.Errorf("no operation specified")
  125. return
  126. }
  127. changedConfig = false
  128. op = "yogurt"
  129. for _, arg := range os.Args[1:] {
  130. if len(arg) < 2 {
  131. continue
  132. }
  133. if arg[0] == '-' && arg[1] != '-' {
  134. switch arg {
  135. case "-V":
  136. arg = "--version"
  137. case "-h":
  138. arg = "--help"
  139. default:
  140. op = arg
  141. continue
  142. }
  143. }
  144. if strings.HasPrefix(arg, "--") {
  145. changedConfig = true
  146. switch arg {
  147. case "--afterclean":
  148. config.CleanAfter = true
  149. case "--noafterclean":
  150. config.CleanAfter = false
  151. case "--printconfig":
  152. fmt.Printf("%#v", config)
  153. os.Exit(0)
  154. case "--gendb":
  155. err = createDevelDB()
  156. if err != nil {
  157. fmt.Println(err)
  158. }
  159. err = saveVCSInfo()
  160. if err != nil {
  161. fmt.Println(err)
  162. }
  163. os.Exit(0)
  164. case "--devel":
  165. config.Devel = true
  166. case "--nodevel":
  167. config.Devel = false
  168. case "--timeupdate":
  169. config.TimeUpdate = true
  170. case "--notimeupdate":
  171. config.TimeUpdate = false
  172. case "--topdown":
  173. config.SortMode = TopDown
  174. case "--complete":
  175. config.Shell = "sh"
  176. _ = complete()
  177. os.Exit(0)
  178. case "--fcomplete":
  179. config.Shell = fishShell
  180. _ = complete()
  181. os.Exit(0)
  182. case "--help":
  183. usage()
  184. os.Exit(0)
  185. case "--version":
  186. fmt.Printf("yay v%s\n", version)
  187. os.Exit(0)
  188. case "--noconfirm":
  189. config.NoConfirm = true
  190. fallthrough
  191. default:
  192. options = append(options, arg)
  193. }
  194. continue
  195. }
  196. packages = append(packages, arg)
  197. }
  198. return
  199. }
  200. func main() {
  201. op, options, pkgs, changedConfig, err := parser()
  202. if err != nil {
  203. fmt.Println(err)
  204. os.Exit(1)
  205. }
  206. switch op {
  207. case "-Cd":
  208. err = cleanDependencies(pkgs)
  209. case "-G":
  210. for _, pkg := range pkgs {
  211. err = getPkgbuild(pkg)
  212. if err != nil {
  213. fmt.Println(pkg+":", err)
  214. }
  215. }
  216. case "-Qstats":
  217. err = localStatistics()
  218. case "-Ss", "-Ssq", "-Sqs":
  219. if op == "-Ss" {
  220. config.SearchMode = Detailed
  221. } else {
  222. config.SearchMode = Minimal
  223. }
  224. if pkgs != nil {
  225. err = syncSearch(pkgs)
  226. }
  227. case "-S":
  228. err = install(pkgs, options)
  229. case "-Sy":
  230. err = passToPacman("-Sy", nil, nil)
  231. if err != nil {
  232. break
  233. }
  234. err = install(pkgs, options)
  235. case "-Syu", "-Suy", "-Su":
  236. if strings.Contains(op, "y") {
  237. err = passToPacman("-Sy", nil, nil)
  238. if err != nil {
  239. break
  240. }
  241. }
  242. err = upgradePkgs(options)
  243. case "-Si":
  244. err = syncInfo(pkgs, options)
  245. case "yogurt":
  246. config.SearchMode = NumberMenu
  247. if pkgs != nil {
  248. err = numberMenu(pkgs, options)
  249. }
  250. default:
  251. if op[0] == 'R' {
  252. removeVCSPackage(pkgs)
  253. }
  254. err = passToPacman(op, pkgs, options)
  255. }
  256. var erra error
  257. if updated {
  258. erra = saveVCSInfo()
  259. if erra != nil {
  260. fmt.Println(err)
  261. }
  262. }
  263. if changedConfig {
  264. erra = config.saveConfig()
  265. if erra != nil {
  266. fmt.Println(err)
  267. }
  268. }
  269. erra = alpmHandle.Release()
  270. if erra != nil {
  271. fmt.Println(err)
  272. }
  273. if err != nil {
  274. fmt.Println(err)
  275. os.Exit(1)
  276. }
  277. }
  278. // NumberMenu presents a CLI for selecting packages to install.
  279. func numberMenu(pkgS []string, flags []string) (err error) {
  280. var num int
  281. aq, err := narrowSearch(pkgS, true)
  282. if err != nil {
  283. fmt.Println("Error during AUR search:", err)
  284. }
  285. numaq := len(aq)
  286. pq, numpq, err := queryRepo(pkgS)
  287. if err != nil {
  288. return
  289. }
  290. if numpq == 0 && numaq == 0 {
  291. return fmt.Errorf("no packages match search")
  292. }
  293. if config.SortMode == BottomUp {
  294. aq.printSearch(numpq)
  295. pq.printSearch()
  296. } else {
  297. pq.printSearch()
  298. aq.printSearch(numpq)
  299. }
  300. fmt.Printf("\x1b[32m%s\x1b[0m\nNumbers: ", "Type numbers to install. Separate each number with a space.")
  301. reader := bufio.NewReader(os.Stdin)
  302. numberBuf, overflow, err := reader.ReadLine()
  303. if err != nil || overflow {
  304. fmt.Println(err)
  305. return
  306. }
  307. numberString := string(numberBuf)
  308. var aurI []string
  309. var repoI []string
  310. result := strings.Fields(numberString)
  311. for _, numS := range result {
  312. num, err = strconv.Atoi(numS)
  313. if err != nil {
  314. continue
  315. }
  316. // Install package
  317. if num > numaq+numpq-1 || num < 0 {
  318. continue
  319. } else if num > numpq-1 {
  320. if config.SortMode == BottomUp {
  321. aurI = append(aurI, aq[numaq+numpq-num-1].Name)
  322. } else {
  323. aurI = append(aurI, aq[num-numpq].Name)
  324. }
  325. } else {
  326. if config.SortMode == BottomUp {
  327. repoI = append(repoI, pq[numpq-num-1].Name())
  328. } else {
  329. repoI = append(repoI, pq[num].Name())
  330. }
  331. }
  332. }
  333. if len(repoI) != 0 {
  334. err = passToPacman("-S", repoI, flags)
  335. }
  336. if len(aurI) != 0 {
  337. err = aurInstall(aurI, flags)
  338. }
  339. return err
  340. }
  341. // Complete provides completion info for shells
  342. func complete() error {
  343. path := completionFile + config.Shell + ".cache"
  344. if info, err := os.Stat(path); os.IsNotExist(err) || time.Since(info.ModTime()).Hours() > 48 {
  345. os.MkdirAll(filepath.Dir(completionFile), 0755)
  346. out, errf := os.Create(path)
  347. if errf != nil {
  348. return errf
  349. }
  350. if createAURList(out) != nil {
  351. defer os.Remove(path)
  352. }
  353. erra := createRepoList(out)
  354. out.Close()
  355. return erra
  356. }
  357. in, err := os.OpenFile(path, os.O_RDWR|os.O_CREATE, 0644)
  358. if err != nil {
  359. return err
  360. }
  361. defer in.Close()
  362. _, err = io.Copy(os.Stdout, in)
  363. return err
  364. }