form.go 2.4 KB

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