amp.go 2.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  1. package main
  2. import (
  3. "context"
  4. "log"
  5. "net/http"
  6. "strings"
  7. "time"
  8. "gitlab.torproject.org/tpo/anti-censorship/pluggable-transports/snowflake/v2/common/amp"
  9. "gitlab.torproject.org/tpo/anti-censorship/pluggable-transports/snowflake/v2/common/messages"
  10. "gitlab.torproject.org/tpo/anti-censorship/pluggable-transports/snowflake/v2/common/util"
  11. )
  12. // ampClientOffers is the AMP-speaking endpoint for client poll messages,
  13. // intended for access via an AMP cache. In contrast to the other clientOffers,
  14. // the client's encoded poll message is stored in the URL path rather than the
  15. // HTTP request body (because an AMP cache does not support POST), and the
  16. // encoded client poll response is sent back as AMP-armored HTML.
  17. func ampClientOffers(i *IPC, w http.ResponseWriter, r *http.Request) {
  18. ctx, cancel := context.WithTimeout(r.Context(), ClientTimeout*time.Second)
  19. defer cancel()
  20. // The encoded client poll message immediately follows the /amp/client/
  21. // path prefix, so this function unfortunately needs to be aware of and
  22. // remote its own routing prefix.
  23. path := strings.TrimPrefix(r.URL.Path, "/amp/client/")
  24. if path == r.URL.Path {
  25. // The path didn't start with the expected prefix. This probably
  26. // indicates an internal bug.
  27. log.Println("ampClientOffers: unexpected prefix in path")
  28. w.WriteHeader(http.StatusInternalServerError)
  29. return
  30. }
  31. var encPollReq []byte
  32. var response []byte
  33. var err error
  34. encPollReq, err = amp.DecodePath(path)
  35. if err == nil {
  36. arg := messages.Arg{
  37. Body: encPollReq,
  38. RemoteAddr: util.GetClientIp(r),
  39. RendezvousMethod: messages.RendezvousAmpCache,
  40. Context: ctx,
  41. }
  42. err = i.ClientOffers(arg, &response)
  43. } else {
  44. response, err = (&messages.ClientPollResponse{
  45. Error: "cannot decode URL path",
  46. }).EncodePollResponse()
  47. }
  48. if err != nil {
  49. // We couldn't even construct a JSON object containing an error
  50. // message :( Nothing to do but signal an error at the HTTP
  51. // layer. The AMP cache will translate this 500 status into a
  52. // 404 status.
  53. // https://amp.dev/documentation/guides-and-tutorials/learn/amp-caches-and-cors/amp-cache-urls/#redirect-%26-error-handling
  54. log.Printf("ampClientOffers: %v", err)
  55. w.WriteHeader(http.StatusInternalServerError)
  56. return
  57. }
  58. w.Header().Set("Content-Type", "text/html")
  59. // Attempt to hint to an AMP cache not to waste resources caching this
  60. // document. "The Google AMP Cache considers any document fresh for at
  61. // least 15 seconds."
  62. // https://developers.google.com/amp/cache/overview#google-amp-cache-updates
  63. w.Header().Set("Cache-Control", "max-age=15")
  64. w.WriteHeader(http.StatusOK)
  65. enc, err := amp.NewArmorEncoder(w)
  66. if err != nil {
  67. log.Printf("amp.NewArmorEncoder: %v", err)
  68. return
  69. }
  70. defer enc.Close()
  71. if _, err := enc.Write(response); err != nil {
  72. log.Printf("ampClientOffers: unable to write answer: %v", err)
  73. }
  74. }