/backend/pastee.go

https://github.com/msparks/pastee · Go · 211 lines · 167 code · 30 blank · 14 comment · 35 complexity · 5e358c02d3ff2689f1d3fb44f29166e3 MD5 · raw file

  1. package pastee
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "net/http"
  7. "regexp"
  8. "strings"
  9. "time"
  10. "appengine"
  11. "appengine/datastore"
  12. )
  13. type PastesGetResp struct {
  14. Content string `json:"content"`
  15. Mac string `json:"mac"`
  16. Expiry string `json:"expiry"`
  17. }
  18. type PastesPostReq struct {
  19. Content string
  20. Mac string
  21. Expiry string
  22. }
  23. type PastesPostResp struct {
  24. Id string `json:"id"`
  25. }
  26. type Paste struct {
  27. Content string
  28. Mac string
  29. Expiry time.Time
  30. }
  31. type ErrorResp struct {
  32. Error string `json:"error"`
  33. }
  34. func init() {
  35. http.HandleFunc("/", indexHandler)
  36. http.HandleFunc("/pastes/", pastesGetHandler)
  37. http.HandleFunc("/pastes", pastesPostHandler)
  38. }
  39. func indexHandler(w http.ResponseWriter, request *http.Request) {
  40. fmt.Fprintf(w, "Index handler")
  41. }
  42. func respondWithError(code int, response ErrorResp, w http.ResponseWriter) {
  43. w.WriteHeader(code)
  44. responseBytes, _ := json.Marshal(response)
  45. fmt.Fprintf(w, "%v\n", string(responseBytes))
  46. }
  47. // Handles GET requests to /pastes/{id}.
  48. func pastesGetHandler(w http.ResponseWriter, r *http.Request) {
  49. if r.Method != "GET" {
  50. w.WriteHeader(http.StatusMethodNotAllowed)
  51. return
  52. }
  53. // Extract MBase31 ID from URL.
  54. mb31IDString := strings.Replace(r.URL.Path, "/pastes/", "", -1)
  55. // Decode MBase31 ID.
  56. // The zeroth ID is special and is considered invalid.
  57. mb31ID, err := MBase31FromString(mb31IDString)
  58. if err != nil || mb31ID.Value == 0 {
  59. // All parse errors result in a 404.
  60. respondWithError(http.StatusNotFound, ErrorResp{Error: "not found"}, w)
  61. return
  62. }
  63. ctx := appengine.NewContext(r)
  64. key := datastore.NewKey(ctx, "paste", "", mb31ID.Value, nil)
  65. var paste Paste
  66. if err := datastore.Get(ctx, key, &paste); err != nil {
  67. respondWithError(http.StatusNotFound, ErrorResp{Error: "not found"}, w)
  68. return
  69. }
  70. // Has the Paste expired?
  71. now := time.Now()
  72. if paste.Expiry.Before(now) {
  73. respondWithError(http.StatusNotFound, ErrorResp{Error: "not found"}, w)
  74. return
  75. }
  76. // Convert Paste to a PasteGetResp.
  77. response := PastesGetResp{}
  78. response.Content = paste.Content
  79. response.Mac = paste.Mac
  80. response.Expiry = paste.Expiry.Format(time.RFC3339)
  81. responseBytes, _ := json.Marshal(response)
  82. fmt.Fprintf(w, "%v\n", string(responseBytes))
  83. }
  84. func pastesPostHandler(w http.ResponseWriter, r *http.Request) {
  85. if r.Method != "POST" {
  86. w.WriteHeader(http.StatusMethodNotAllowed)
  87. return
  88. }
  89. const kMaxBodyLength = 256 * 1024
  90. if r.ContentLength > kMaxBodyLength {
  91. respondWithError(
  92. http.StatusRequestEntityTooLarge,
  93. ErrorResp{Error: "body too large"}, w)
  94. return
  95. }
  96. if r.ContentLength < 0 {
  97. respondWithError(
  98. http.StatusLengthRequired,
  99. ErrorResp{Error: "content length required"}, w)
  100. return
  101. }
  102. postData := make([]byte, r.ContentLength)
  103. _, err := r.Body.Read(postData)
  104. if err != nil {
  105. respondWithError(
  106. http.StatusBadRequest,
  107. ErrorResp{Error: "POST body required"}, w)
  108. return
  109. }
  110. var request PastesPostReq
  111. err = json.Unmarshal(postData, &request)
  112. if err != nil {
  113. respondWithError(
  114. http.StatusBadRequest,
  115. ErrorResp{Error: err.Error()}, w)
  116. return
  117. }
  118. ctx := appengine.NewContext(r)
  119. code, response, err := pastesPostRPC(&ctx, &request)
  120. w.WriteHeader(code)
  121. var responseBytes []byte
  122. if err != nil {
  123. respondWithError(code, ErrorResp{Error: err.Error()}, w)
  124. } else {
  125. responseBytes, _ = json.Marshal(response)
  126. fmt.Fprintf(w, "%v\n", string(responseBytes))
  127. }
  128. }
  129. func pastesPostRPC(ctx *appengine.Context, request *PastesPostReq) (int, PastesPostResp, error) {
  130. // TODO(ms): These should be configurable.
  131. const kMaxContentLength = 256 * 1024 // 256 KiB
  132. const kMaxMacLength = 128
  133. const kMaxLifetime = 7 * 24 * time.Hour
  134. const kDefaultLifetime = 1 * time.Hour
  135. // Length checking.
  136. if request.Content == "" {
  137. return http.StatusBadRequest, PastesPostResp{}, errors.New("content is required")
  138. } else if len(request.Content) > kMaxContentLength {
  139. return http.StatusBadRequest, PastesPostResp{}, errors.New(
  140. fmt.Sprintf("max content is %d bytes", kMaxContentLength))
  141. } else if len(request.Mac) > kMaxMacLength {
  142. return http.StatusBadRequest, PastesPostResp{}, errors.New(
  143. fmt.Sprintf("max mac length is %d bytes", kMaxMacLength))
  144. }
  145. // Verify MAC looks lke a hex digest.
  146. if m, _ := regexp.MatchString("^[a-f0-9]*$", request.Mac); !m {
  147. return http.StatusBadRequest, PastesPostResp{}, errors.New(
  148. fmt.Sprintf("mac must be a hex digest"))
  149. }
  150. // Parse and validate expiration date.
  151. now := time.Now()
  152. var expiry time.Time
  153. if request.Expiry != "" {
  154. var err error
  155. expiry, err = time.Parse(time.RFC3339, request.Expiry)
  156. if err != nil {
  157. return http.StatusBadRequest, PastesPostResp{}, errors.New("bad time format")
  158. }
  159. if expiry.After(now.Add(kMaxLifetime)) {
  160. return http.StatusBadRequest, PastesPostResp{}, errors.New(
  161. fmt.Sprintf("maximum lifetime is %v", kMaxLifetime))
  162. }
  163. } else {
  164. expiry = now.Add(kDefaultLifetime)
  165. }
  166. // Construct Paste entity for datastore.
  167. var paste Paste
  168. paste.Content = request.Content
  169. paste.Mac = request.Mac
  170. paste.Expiry = expiry
  171. // Insert Paste.
  172. key, err := datastore.Put(
  173. *ctx, datastore.NewIncompleteKey(*ctx, "paste", nil), &paste)
  174. if err != nil {
  175. return http.StatusInternalServerError, PastesPostResp{}, err
  176. }
  177. // Paste created successfully.
  178. var response PastesPostResp
  179. response.Id = MBase31{Value: key.IntID()}.ToString()
  180. return http.StatusCreated, response, nil
  181. }