completion.go 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. package completion
  2. import (
  3. "bufio"
  4. "context"
  5. "fmt"
  6. "io"
  7. "net/http"
  8. "net/url"
  9. "os"
  10. "path"
  11. "path/filepath"
  12. "strings"
  13. "time"
  14. "github.com/Jguer/yay/v12/pkg/db"
  15. )
  16. type PkgSynchronizer interface {
  17. SyncPackages(...string) []db.IPackage
  18. }
  19. type httpRequestDoer interface {
  20. Do(req *http.Request) (*http.Response, error)
  21. }
  22. // Show provides completion info for shells.
  23. func Show(ctx context.Context, httpClient httpRequestDoer,
  24. dbExecutor PkgSynchronizer, aurURL, completionPath string, interval int, force bool,
  25. ) error {
  26. err := Update(ctx, httpClient, dbExecutor, aurURL, completionPath, interval, force)
  27. if err != nil {
  28. return err
  29. }
  30. in, err := os.OpenFile(completionPath, os.O_RDWR|os.O_CREATE, 0o644)
  31. if err != nil {
  32. return err
  33. }
  34. defer in.Close()
  35. _, err = io.Copy(os.Stdout, in)
  36. return err
  37. }
  38. // Update updates completion cache to be used by Complete.
  39. func Update(ctx context.Context, httpClient httpRequestDoer,
  40. dbExecutor PkgSynchronizer, aurURL, completionPath string, interval int, force bool,
  41. ) error {
  42. info, err := os.Stat(completionPath)
  43. if os.IsNotExist(err) || (interval != -1 && time.Since(info.ModTime()).Hours() >= float64(interval*24)) || force {
  44. errd := os.MkdirAll(filepath.Dir(completionPath), 0o755)
  45. if errd != nil {
  46. return errd
  47. }
  48. out, errf := os.Create(completionPath)
  49. if errf != nil {
  50. return errf
  51. }
  52. if createAURList(ctx, httpClient, aurURL, out) != nil {
  53. defer os.Remove(completionPath)
  54. }
  55. erra := createRepoList(dbExecutor, out)
  56. out.Close()
  57. return erra
  58. }
  59. return nil
  60. }
  61. // CreateAURList creates a new completion file.
  62. func createAURList(ctx context.Context, client httpRequestDoer, aurURL string, out io.Writer) error {
  63. u, err := url.Parse(aurURL)
  64. if err != nil {
  65. return err
  66. }
  67. u.Path = path.Join(u.Path, "packages.gz")
  68. req, err := http.NewRequestWithContext(ctx, http.MethodGet, u.String(), http.NoBody)
  69. if err != nil {
  70. return err
  71. }
  72. resp, err := client.Do(req)
  73. if err != nil {
  74. return err
  75. }
  76. defer resp.Body.Close()
  77. if resp.StatusCode != http.StatusOK {
  78. return fmt.Errorf("invalid status code: %d", resp.StatusCode)
  79. }
  80. scanner := bufio.NewScanner(resp.Body)
  81. scanner.Scan()
  82. for scanner.Scan() {
  83. text := scanner.Text()
  84. if strings.HasPrefix(text, "#") {
  85. continue
  86. }
  87. if _, err := io.WriteString(out, text+"\tAUR\n"); err != nil {
  88. return err
  89. }
  90. }
  91. return nil
  92. }
  93. // createRepoList appends Repo packages to completion cache.
  94. func createRepoList(dbExecutor PkgSynchronizer, out io.Writer) error {
  95. for _, pkg := range dbExecutor.SyncPackages() {
  96. _, err := io.WriteString(out, pkg.Name()+"\t"+pkg.DB().Name()+"\n")
  97. if err != nil {
  98. return err
  99. }
  100. }
  101. return nil
  102. }