api.go 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. package web
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "net/http"
  7. "net/url"
  8. "time"
  9. "github.com/kellegous/go/context"
  10. "github.com/syndtr/goleveldb/leveldb"
  11. )
  12. const (
  13. alpha = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
  14. )
  15. var (
  16. errInvalidURL = errors.New("Invalid URL")
  17. errRedirectLoop = errors.New(" I'm sorry, Dave. I'm afraid I can't do that")
  18. )
  19. // A very simple encoding of numeric ids. This is simply a base62 encoding
  20. // prefixed with ":"
  21. func encodeID(id uint64) string {
  22. n := uint64(len(alpha))
  23. b := make([]byte, 0, 8)
  24. if id == 0 {
  25. return "0"
  26. }
  27. b = append(b, ':')
  28. for id > 0 {
  29. b = append(b, alpha[id%n])
  30. id /= n
  31. }
  32. return string(b)
  33. }
  34. // Check that the given URL is suitable as a shortcut link.
  35. func validateURL(r *http.Request, s string) error {
  36. u, err := url.Parse(s)
  37. if err != nil {
  38. return errInvalidURL
  39. }
  40. switch u.Scheme {
  41. case "http", "https", "mailto", "ftp":
  42. break
  43. default:
  44. return errInvalidURL
  45. }
  46. if r.Host == u.Host {
  47. return errRedirectLoop
  48. }
  49. return nil
  50. }
  51. func apiURLPost(ctx *context.Context, w http.ResponseWriter, r *http.Request) {
  52. p := parseName("/api/url/", r.URL.Path)
  53. var req struct {
  54. URL string `json:"url"`
  55. }
  56. if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
  57. writeJSONError(w, "invalid json")
  58. return
  59. }
  60. // Handle delete requests
  61. if req.URL == "" {
  62. if p == "" {
  63. writeJSONError(w, "url required")
  64. return
  65. }
  66. if err := ctx.Del(p); err != nil {
  67. writeJSONBackendError(w, err)
  68. return
  69. }
  70. writeJSONOk(w)
  71. return
  72. }
  73. if err := validateURL(r, req.URL); err != nil {
  74. writeJSONError(w, err.Error())
  75. return
  76. }
  77. // If no name is specified, an ID must be generate.
  78. if p == "" {
  79. id, err := ctx.NextID()
  80. if err != nil {
  81. writeJSONBackendError(w, err)
  82. return
  83. }
  84. p = encodeID(id)
  85. }
  86. rt := context.Route{
  87. URL: req.URL,
  88. Time: time.Now(),
  89. }
  90. if err := ctx.Put(p, &rt); err != nil {
  91. writeJSONBackendError(w, err)
  92. return
  93. }
  94. writeJSONRoute(w, p, &rt)
  95. }
  96. func apiURLGet(ctx *context.Context, w http.ResponseWriter, r *http.Request) {
  97. p := parseName("/api/url/", r.URL.Path)
  98. if p == "" {
  99. writeJSONOk(w)
  100. return
  101. }
  102. rt, err := ctx.Get(p)
  103. if err == leveldb.ErrNotFound {
  104. writeJSONOk(w)
  105. return
  106. } else if err != nil {
  107. writeJSONBackendError(w, err)
  108. return
  109. }
  110. writeJSONRoute(w, p, rt)
  111. }
  112. func apiURLDelete(ctx *context.Context, w http.ResponseWriter, r *http.Request) {
  113. p := parseName("/api/url/", r.URL.Path)
  114. if p == "" {
  115. writeJSONError(w, "name required")
  116. return
  117. }
  118. if err := ctx.Del(p); err == leveldb.ErrNotFound {
  119. writeJSONError(w, fmt.Sprintf("%s not found", p))
  120. return
  121. } else if err != nil {
  122. writeJSONBackendError(w, err)
  123. return
  124. }
  125. writeJSONOk(w)
  126. }
  127. func apiURLsGet(ctx *context.Context, w http.ResponseWriter, r *http.Request) {
  128. // TODO(knorton): This will allow enumeration of the routes.
  129. writeJSONError(w, http.StatusText(http.StatusNotImplemented))
  130. }
  131. func apiURL(ctx *context.Context, w http.ResponseWriter, r *http.Request) {
  132. switch r.Method {
  133. case "POST":
  134. apiURLPost(ctx, w, r)
  135. case "GET":
  136. apiURLGet(ctx, w, r)
  137. case "DELETE":
  138. apiURLDelete(ctx, w, r)
  139. default:
  140. writeJSONError(w, http.StatusText(http.StatusMethodNotAllowed))
  141. }
  142. }
  143. func apiURLs(ctx *context.Context, w http.ResponseWriter, r *http.Request) {
  144. switch r.Method {
  145. case "GET":
  146. apiURLsGet(ctx, w, r)
  147. default:
  148. writeJSONError(w, http.StatusText(http.StatusMethodNotAllowed))
  149. }
  150. }
  151. // Setup ...
  152. func Setup(m *http.ServeMux, ctx *context.Context) {
  153. m.HandleFunc("/api/url/", func(w http.ResponseWriter, r *http.Request) {
  154. apiURL(ctx, w, r)
  155. })
  156. m.HandleFunc("/api/urls/", func(w http.ResponseWriter, r *http.Request) {
  157. apiURLs(ctx, w, r)
  158. })
  159. }