cmd.go 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410
  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 "--bottomup":
  175. config.SortMode = BottomUp
  176. case "--complete":
  177. config.Shell = "sh"
  178. _ = complete()
  179. os.Exit(0)
  180. case "--fcomplete":
  181. config.Shell = fishShell
  182. _ = complete()
  183. os.Exit(0)
  184. case "--help":
  185. usage()
  186. os.Exit(0)
  187. case "--version":
  188. fmt.Printf("yay v%s\n", version)
  189. os.Exit(0)
  190. case "--noconfirm":
  191. config.NoConfirm = true
  192. fallthrough
  193. default:
  194. options = append(options, arg)
  195. }
  196. continue
  197. }
  198. packages = append(packages, arg)
  199. }
  200. return
  201. }
  202. func main() {
  203. op, options, pkgs, changedConfig, err := parser()
  204. if err != nil {
  205. fmt.Println(err)
  206. os.Exit(1)
  207. }
  208. switch op {
  209. case "-Cd":
  210. err = cleanDependencies(pkgs)
  211. case "-G":
  212. for _, pkg := range pkgs {
  213. err = getPkgbuild(pkg)
  214. if err != nil {
  215. fmt.Println(pkg+":", err)
  216. }
  217. }
  218. case "-Qstats":
  219. err = localStatistics()
  220. case "-Ss", "-Ssq", "-Sqs":
  221. if op == "-Ss" {
  222. config.SearchMode = Detailed
  223. } else {
  224. config.SearchMode = Minimal
  225. }
  226. if pkgs != nil {
  227. err = syncSearch(pkgs)
  228. }
  229. case "-S":
  230. err = install(pkgs, options)
  231. case "-Sy":
  232. err = passToPacman("-Sy", nil, nil)
  233. if err != nil {
  234. break
  235. }
  236. err = install(pkgs, options)
  237. case "-Syu", "-Suy", "-Su":
  238. if strings.Contains(op, "y") {
  239. err = passToPacman("-Sy", nil, nil)
  240. if err != nil {
  241. break
  242. }
  243. }
  244. err = upgradePkgs(options)
  245. case "-Si":
  246. err = syncInfo(pkgs, options)
  247. case "yogurt":
  248. config.SearchMode = NumberMenu
  249. if pkgs != nil {
  250. err = numberMenu(pkgs, options)
  251. }
  252. default:
  253. if op[0] == 'R' {
  254. removeVCSPackage(pkgs)
  255. }
  256. err = passToPacman(op, pkgs, options)
  257. }
  258. var erra error
  259. if updated {
  260. erra = saveVCSInfo()
  261. if erra != nil {
  262. fmt.Println(err)
  263. }
  264. }
  265. if changedConfig {
  266. erra = config.saveConfig()
  267. if erra != nil {
  268. fmt.Println(err)
  269. }
  270. }
  271. erra = alpmHandle.Release()
  272. if erra != nil {
  273. fmt.Println(err)
  274. }
  275. if err != nil {
  276. fmt.Println(err)
  277. os.Exit(1)
  278. }
  279. }
  280. // NumberMenu presents a CLI for selecting packages to install.
  281. func numberMenu(pkgS []string, flags []string) (err error) {
  282. var num int
  283. aq, err := narrowSearch(pkgS, true)
  284. if err != nil {
  285. fmt.Println("Error during AUR search:", err)
  286. }
  287. numaq := len(aq)
  288. pq, numpq, err := queryRepo(pkgS)
  289. if err != nil {
  290. return
  291. }
  292. if numpq == 0 && numaq == 0 {
  293. return fmt.Errorf("no packages match search")
  294. }
  295. if config.SortMode == BottomUp {
  296. aq.printSearch(numpq)
  297. pq.printSearch()
  298. } else {
  299. pq.printSearch()
  300. aq.printSearch(numpq)
  301. }
  302. fmt.Printf("\x1b[32m%s\x1b[0m\nNumbers: ", "Type numbers to install. Separate each number with a space.")
  303. reader := bufio.NewReader(os.Stdin)
  304. numberBuf, overflow, err := reader.ReadLine()
  305. if err != nil || overflow {
  306. fmt.Println(err)
  307. return
  308. }
  309. numberString := string(numberBuf)
  310. var aurI []string
  311. var repoI []string
  312. result := strings.Fields(numberString)
  313. for _, numS := range result {
  314. num, err = strconv.Atoi(numS)
  315. if err != nil {
  316. continue
  317. }
  318. // Install package
  319. if num > numaq+numpq-1 || num < 0 {
  320. continue
  321. } else if num > numpq-1 {
  322. if config.SortMode == BottomUp {
  323. aurI = append(aurI, aq[numaq+numpq-num-1].Name)
  324. } else {
  325. aurI = append(aurI, aq[num-numpq].Name)
  326. }
  327. } else {
  328. if config.SortMode == BottomUp {
  329. repoI = append(repoI, pq[numpq-num-1].Name())
  330. } else {
  331. repoI = append(repoI, pq[num].Name())
  332. }
  333. }
  334. }
  335. if len(repoI) != 0 {
  336. err = passToPacman("-S", repoI, flags)
  337. }
  338. if len(aurI) != 0 {
  339. err = aurInstall(aurI, flags)
  340. }
  341. return err
  342. }
  343. // Complete provides completion info for shells
  344. func complete() error {
  345. path := completionFile + config.Shell + ".cache"
  346. if info, err := os.Stat(path); os.IsNotExist(err) || time.Since(info.ModTime()).Hours() > 48 {
  347. os.MkdirAll(filepath.Dir(completionFile), 0755)
  348. out, errf := os.Create(path)
  349. if errf != nil {
  350. return errf
  351. }
  352. if createAURList(out) != nil {
  353. defer os.Remove(path)
  354. }
  355. erra := createRepoList(out)
  356. out.Close()
  357. return erra
  358. }
  359. in, err := os.OpenFile(path, os.O_RDWR|os.O_CREATE, 0644)
  360. if err != nil {
  361. return err
  362. }
  363. defer in.Close()
  364. _, err = io.Copy(os.Stdout, in)
  365. return err
  366. }