form.go 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. package login
  2. import (
  3. "errors"
  4. "log/slog"
  5. "net/http"
  6. "github.com/ayn2op/discordo/internal/config"
  7. "github.com/ayn2op/discordo/internal/consts"
  8. "github.com/ayn2op/discordo/internal/ui"
  9. "github.com/diamondburned/arikawa/v3/api"
  10. "github.com/diamondburned/arikawa/v3/utils/httputil"
  11. "github.com/rivo/tview"
  12. "github.com/zalando/go-keyring"
  13. )
  14. type DoneFn = func(token string)
  15. type Form struct {
  16. *tview.Pages
  17. cfg *config.Config
  18. app *tview.Application
  19. form *tview.Form
  20. done DoneFn
  21. }
  22. func NewForm(cfg *config.Config, app *tview.Application, done DoneFn) *Form {
  23. f := &Form{
  24. Pages: tview.NewPages(),
  25. cfg: cfg,
  26. app: app,
  27. form: tview.NewForm(),
  28. done: done,
  29. }
  30. f.form.
  31. AddInputField("Email", "", 0, nil, nil).
  32. AddPasswordField("Password", "", 0, 0, nil).
  33. AddPasswordField("Code (optional)", "", 0, 0, nil).
  34. AddButton("Login", f.login)
  35. f.AddAndSwitchToPage("form", f.form, true)
  36. return f
  37. }
  38. func (f *Form) login() {
  39. email := f.form.GetFormItem(0).(*tview.InputField).GetText()
  40. password := f.form.GetFormItem(1).(*tview.InputField).GetText()
  41. if email == "" || password == "" {
  42. return
  43. }
  44. // Create an API client without an authentication token.
  45. client := api.NewClient("")
  46. // Spoof the user agent of a web browser.
  47. client.UserAgent = f.cfg.Identify.UserAgent
  48. body := httputil.WithJSONBody(struct {
  49. Email string `json:"login"`
  50. Password string `json:"password"`
  51. }{email, password})
  52. var (
  53. resp *api.LoginResponse
  54. err error
  55. )
  56. err = client.RequestJSON(&resp, http.MethodPost, api.EndpointLogin, body)
  57. if err != nil {
  58. f.onError(err)
  59. return
  60. }
  61. if resp.Token == "" && resp.MFA {
  62. code := f.form.GetFormItem(2).(*tview.InputField).GetText()
  63. if code == "" {
  64. f.onError(errors.New("code required"))
  65. return
  66. }
  67. // Attempt to login using the code.
  68. resp, err = client.TOTP(code, resp.Ticket)
  69. if err != nil {
  70. f.onError(err)
  71. return
  72. }
  73. }
  74. if resp.Token == "" {
  75. f.onError(errors.New("missing token"))
  76. return
  77. }
  78. go keyring.Set(consts.Name, "token", resp.Token)
  79. if f.done != nil {
  80. f.done(resp.Token)
  81. }
  82. }
  83. func (f *Form) onError(err error) {
  84. slog.Error("failed to login", "err", err)
  85. modal := tview.NewModal().
  86. SetText(err.Error()).
  87. AddButtons([]string{"Close"}).
  88. SetDoneFunc(func(_ int, _ string) {
  89. f.RemovePage("modal").SwitchToPage("form")
  90. })
  91. f.
  92. AddAndSwitchToPage("modal", ui.Centered(modal, 0, 0), true).
  93. ShowPage("form")
  94. }