A webmail client. Forked from https://git.sr.ht/~migadu/alps
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

170 lines
4.2 KiB

  1. package alpscarddav
  2. import (
  3. "fmt"
  4. "net/http"
  5. "net/url"
  6. "strings"
  7. "git.sr.ht/~emersion/alps"
  8. alpsbase "git.sr.ht/~emersion/alps/plugins/base"
  9. "github.com/emersion/go-vcard"
  10. "github.com/emersion/go-webdav/carddav"
  11. )
  12. func sanityCheckURL(u *url.URL) error {
  13. req, err := http.NewRequest(http.MethodOptions, u.String(), nil)
  14. if err != nil {
  15. return err
  16. }
  17. resp, err := http.DefaultClient.Do(req)
  18. if err != nil {
  19. return err
  20. }
  21. resp.Body.Close()
  22. // Servers might require authentication to perform an OPTIONS request
  23. if resp.StatusCode/100 != 2 && resp.StatusCode != http.StatusUnauthorized {
  24. return fmt.Errorf("HTTP request failed: %v %v", resp.StatusCode, resp.Status)
  25. }
  26. return nil
  27. }
  28. type plugin struct {
  29. alps.GoPlugin
  30. url *url.URL
  31. homeSetCache map[string]string
  32. }
  33. func (p *plugin) client(session *alps.Session) (*carddav.Client, error) {
  34. return newClient(p.url, session)
  35. }
  36. func (p *plugin) clientWithAddressBook(session *alps.Session) (*carddav.Client, *carddav.AddressBook, error) {
  37. c, err := newClient(p.url, session)
  38. if err != nil {
  39. return nil, nil, fmt.Errorf("failed to create CardDAV client: %v", err)
  40. }
  41. homeSet, ok := p.homeSetCache[session.Username()]
  42. if !ok {
  43. principal, err := c.FindCurrentUserPrincipal()
  44. if err != nil {
  45. return nil, nil, fmt.Errorf("failed to query CardDAV principal: %v", err)
  46. }
  47. homeSet, err = c.FindAddressBookHomeSet(principal)
  48. if err != nil {
  49. return nil, nil, fmt.Errorf("failed to query CardDAV address book home set: %v", err)
  50. }
  51. p.homeSetCache[session.Username()] = homeSet
  52. // TODO: evict entries from the cache if it's getting too big
  53. }
  54. addressBooks, err := c.FindAddressBooks(homeSet)
  55. if err != nil {
  56. return nil, nil, fmt.Errorf("failed to query CardDAV address books: %v", err)
  57. }
  58. if len(addressBooks) == 0 {
  59. return nil, nil, errNoAddressBook
  60. }
  61. return c, &addressBooks[0], nil
  62. }
  63. func newPlugin(srv *alps.Server) (alps.Plugin, error) {
  64. u, err := srv.Upstream("carddavs", "carddav+insecure", "https", "http+insecure")
  65. if _, ok := err.(*alps.NoUpstreamError); ok {
  66. return nil, nil
  67. } else if err != nil {
  68. return nil, fmt.Errorf("carddav: failed to parse upstream CardDAV server: %v", err)
  69. }
  70. switch u.Scheme {
  71. case "carddavs":
  72. u.Scheme = "https"
  73. case "carddav+insecure", "http+insecure":
  74. u.Scheme = "http"
  75. }
  76. if u.Scheme == "" {
  77. s, err := carddav.Discover(u.Host)
  78. if err != nil {
  79. srv.Logger().Printf("carddav: failed to discover CardDAV server: %v", err)
  80. return nil, nil
  81. }
  82. u, err = url.Parse(s)
  83. if err != nil {
  84. return nil, fmt.Errorf("carddav: Discover returned an invalid URL: %v", err)
  85. }
  86. }
  87. if err := sanityCheckURL(u); err != nil {
  88. return nil, fmt.Errorf("carddav: failed to connect to CardDAV server %q: %v", u, err)
  89. }
  90. srv.Logger().Printf("Configured upstream CardDAV server: %v", u)
  91. p := &plugin{
  92. GoPlugin: alps.GoPlugin{Name: "carddav"},
  93. url: u,
  94. homeSetCache: make(map[string]string),
  95. }
  96. registerRoutes(p)
  97. p.TemplateFuncs(map[string]interface{}{
  98. "join": func(l []string, sep string) string {
  99. return strings.Join(l, sep)
  100. },
  101. })
  102. p.Inject("compose.html", func(ctx *alps.Context, _data alps.RenderData) error {
  103. data := _data.(*alpsbase.ComposeRenderData)
  104. c, addressBook, err := p.clientWithAddressBook(ctx.Session)
  105. if err == errNoAddressBook {
  106. return nil
  107. } else if err != nil {
  108. return err
  109. }
  110. query := carddav.AddressBookQuery{
  111. DataRequest: carddav.AddressDataRequest{
  112. Props: []string{vcard.FieldFormattedName, vcard.FieldEmail},
  113. },
  114. PropFilters: []carddav.PropFilter{{
  115. Name: vcard.FieldEmail,
  116. }},
  117. }
  118. addrs, err := c.QueryAddressBook(addressBook.Path, &query)
  119. if err != nil {
  120. return fmt.Errorf("failed to query CardDAV addresses: %v", err)
  121. }
  122. // TODO: cache the results
  123. emails := make([]string, 0, len(addrs))
  124. for _, addr := range addrs {
  125. cardEmails := addr.Card.Values(vcard.FieldEmail)
  126. emails = append(emails, cardEmails...)
  127. }
  128. data.Extra["EmailSuggestions"] = emails
  129. return nil
  130. })
  131. return p.Plugin(), nil
  132. }
  133. func init() {
  134. alps.RegisterPluginLoader(func(s *alps.Server) ([]alps.Plugin, error) {
  135. p, err := newPlugin(s)
  136. if err != nil {
  137. return nil, err
  138. }
  139. if p == nil {
  140. return nil, nil
  141. }
  142. return []alps.Plugin{p}, err
  143. })
  144. }