api_test.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588
  1. package web
  2. import (
  3. "bytes"
  4. "context"
  5. "encoding/base64"
  6. "encoding/json"
  7. "errors"
  8. "fmt"
  9. "io"
  10. "io/ioutil"
  11. "net/http"
  12. "net/url"
  13. "os"
  14. "path/filepath"
  15. "testing"
  16. "time"
  17. "github.com/kellegous/go/backend"
  18. "github.com/kellegous/go/backend/leveldb"
  19. "github.com/kellegous/go/internal"
  20. )
  21. type urlReq struct {
  22. URL string `json:"url"`
  23. }
  24. type env struct {
  25. mux *http.ServeMux
  26. dir string
  27. backend backend.Backend
  28. }
  29. func (e *env) destroy() {
  30. os.RemoveAll(e.dir)
  31. }
  32. func (e *env) get(path string) (*mockResponse, error) {
  33. return e.call("GET", path, nil)
  34. }
  35. func (e *env) post(path string, body interface{}) (*mockResponse, error) {
  36. return e.callWithJSON("POST", path, body)
  37. }
  38. func (e *env) callWithJSON(method, path string, body interface{}) (*mockResponse, error) {
  39. var r io.Reader
  40. if body != nil {
  41. var buf bytes.Buffer
  42. if err := json.NewEncoder(&buf).Encode(body); err != nil {
  43. return nil, err
  44. }
  45. r = &buf
  46. }
  47. return e.call(method, path, r)
  48. }
  49. func (e *env) call(method, path string, body io.Reader) (*mockResponse, error) {
  50. req, err := http.NewRequest(method, path, body)
  51. if err != nil {
  52. return nil, err
  53. }
  54. res := &mockResponse{
  55. header: map[string][]string{},
  56. }
  57. e.mux.ServeHTTP(res, req)
  58. return res, nil
  59. }
  60. func newEnv() (*env, error) {
  61. dir, err := ioutil.TempDir("", "")
  62. if err != nil {
  63. return nil, err
  64. }
  65. backend, err := leveldb.New(filepath.Join(dir, "data"))
  66. if err != nil {
  67. os.RemoveAll(dir)
  68. return nil, err
  69. }
  70. mux := http.NewServeMux()
  71. Setup(mux, backend)
  72. return &env{
  73. mux: mux,
  74. dir: dir,
  75. backend: backend,
  76. }, nil
  77. }
  78. func needEnv(t *testing.T) *env {
  79. e, err := newEnv()
  80. if err != nil {
  81. t.Fatal(err)
  82. }
  83. return e
  84. }
  85. type mockResponse struct {
  86. header http.Header
  87. bytes.Buffer
  88. status int
  89. }
  90. func (r *mockResponse) Header() http.Header {
  91. return r.header
  92. }
  93. func (r *mockResponse) WriteHeader(status int) {
  94. r.status = status
  95. }
  96. func mustBeSameNamedRoute(t *testing.T, a, b *routeWithName) {
  97. if a.Name != b.Name || a.URL != b.URL || a.Time.UnixNano() != b.Time.UnixNano() {
  98. t.Fatalf("routes are not same: %v vs %v", a, b)
  99. }
  100. }
  101. func mustBeRouteOf(t *testing.T, rt *internal.Route, url string) {
  102. if rt == nil {
  103. t.Fatal("route is nil")
  104. }
  105. if rt.URL != url {
  106. t.Fatalf("expected url of %s, got %s", url, rt.URL)
  107. }
  108. if rt.Time.IsZero() {
  109. t.Fatal("route time is empty")
  110. }
  111. }
  112. func mustBeNamedRouteOf(t *testing.T, rt *routeWithName, name, url string) {
  113. mustBeRouteOf(t, rt.Route, url)
  114. if rt.Name != name {
  115. t.Fatalf("expected name of %s, got %s", name, rt.Name)
  116. }
  117. }
  118. func mustBeOk(t *testing.T, ok bool) {
  119. if !ok {
  120. t.Fatal("response is not ok")
  121. }
  122. }
  123. func mustBeErr(t *testing.T, m *msgErr) {
  124. if m.Ok {
  125. t.Fatal("response is ok, should be err")
  126. }
  127. if m.Error == "" {
  128. t.Fatal("expected an Error, but it is empty")
  129. }
  130. }
  131. func mustHaveStatus(t *testing.T, res *mockResponse, status int) {
  132. if res.status != status {
  133. t.Fatalf("expected response status %d, got %d", status, res.status)
  134. }
  135. }
  136. func TestAPIGetNotFound(t *testing.T) {
  137. e := needEnv(t)
  138. defer e.destroy()
  139. names := map[string]int{
  140. "": http.StatusBadRequest,
  141. "nothing": http.StatusNotFound,
  142. "nothing/there": http.StatusNotFound,
  143. }
  144. for name, status := range names {
  145. res, err := e.get(fmt.Sprintf("/api/url/%s", name))
  146. if err != nil {
  147. t.Fatal(err)
  148. }
  149. mustHaveStatus(t, res, status)
  150. var m msgErr
  151. if err := json.NewDecoder(res).Decode(&m); err != nil {
  152. t.Fatal(err)
  153. }
  154. mustBeErr(t, &m)
  155. }
  156. }
  157. func TestAPIPutThenGet(t *testing.T) {
  158. e := needEnv(t)
  159. defer e.destroy()
  160. res, err := e.post("/api/url/xxx", &urlReq{
  161. URL: "http://ex.com/",
  162. })
  163. if err != nil {
  164. t.Fatal(err)
  165. }
  166. mustHaveStatus(t, res, http.StatusOK)
  167. var pm msgRoute
  168. if err := json.NewDecoder(res).Decode(&pm); err != nil {
  169. t.Fatal(err)
  170. }
  171. mustBeOk(t, pm.Ok)
  172. mustBeNamedRouteOf(t, pm.Route, "xxx", "http://ex.com/")
  173. res, err = e.get("/api/url/xxx")
  174. if err != nil {
  175. t.Fatal(err)
  176. }
  177. mustHaveStatus(t, res, http.StatusOK)
  178. var gm msgRoute
  179. if err := json.NewDecoder(res).Decode(&gm); err != nil {
  180. t.Fatal(err)
  181. }
  182. mustBeOk(t, gm.Ok)
  183. mustBeNamedRouteOf(t, pm.Route, "xxx", "http://ex.com/")
  184. }
  185. func TestBadPuts(t *testing.T) {
  186. e := needEnv(t)
  187. defer e.destroy()
  188. var m msgErr
  189. res, err := e.call("POST", "/api/url/yyy", bytes.NewBufferString("not json"))
  190. if err != nil {
  191. t.Fatal(err)
  192. }
  193. mustHaveStatus(t, res, http.StatusBadRequest)
  194. if err := json.NewDecoder(res).Decode(&m); err != nil {
  195. t.Fatal(err)
  196. }
  197. mustBeErr(t, &m)
  198. res, err = e.post("/api/url/yyy", &urlReq{})
  199. if err != nil {
  200. t.Fatal(err)
  201. }
  202. mustHaveStatus(t, res, http.StatusBadRequest)
  203. if err := json.NewDecoder(res).Decode(&m); err != nil {
  204. t.Fatal(err)
  205. }
  206. mustBeErr(t, &m)
  207. res, err = e.post("/api/url/yyy", &urlReq{"not a URL"})
  208. if err != nil {
  209. t.Fatal(err)
  210. }
  211. mustHaveStatus(t, res, http.StatusBadRequest)
  212. if err := json.NewDecoder(res).Decode(&m); err != nil {
  213. t.Fatal(err)
  214. }
  215. mustBeErr(t, &m)
  216. }
  217. func TestAPIDel(t *testing.T) {
  218. e := needEnv(t)
  219. defer e.destroy()
  220. ctx, cancel := context.WithTimeout(context.Background(), time.Minute)
  221. defer cancel()
  222. if err := e.backend.Put(ctx, "xxx", &internal.Route{
  223. URL: "http://ex.com/",
  224. Time: time.Now(),
  225. }); err != nil {
  226. t.Fatal(err)
  227. }
  228. res, err := e.call("DELETE", "/api/url/xxx", nil)
  229. if err != nil {
  230. t.Fatal(err)
  231. }
  232. mustHaveStatus(t, res, http.StatusOK)
  233. var m msg
  234. if err := json.NewDecoder(res).Decode(&m); err != nil {
  235. t.Fatal(err)
  236. }
  237. mustBeOk(t, m.Ok)
  238. ctx, cancel = context.WithTimeout(context.Background(), time.Minute)
  239. defer cancel()
  240. if _, err := e.backend.Get(ctx, "xxx"); !errors.Is(err, internal.ErrRouteNotFound) {
  241. t.Fatal("expected xxx to be deleted")
  242. }
  243. }
  244. func TestAPIPutThenGetAuto(t *testing.T) {
  245. e := needEnv(t)
  246. defer e.destroy()
  247. res, err := e.post("/api/url/", &urlReq{URL: "http://b.com/"})
  248. if err != nil {
  249. t.Fatal(err)
  250. }
  251. mustHaveStatus(t, res, http.StatusOK)
  252. var am msgRoute
  253. if err := json.NewDecoder(res).Decode(&am); err != nil {
  254. t.Fatal(err)
  255. }
  256. mustBeOk(t, am.Ok)
  257. mustBeRouteOf(t, am.Route.Route, "http://b.com/")
  258. res, err = e.get(fmt.Sprintf("/api/url/%s", am.Route.Name))
  259. if err != nil {
  260. t.Fatal(err)
  261. }
  262. mustHaveStatus(t, res, http.StatusOK)
  263. var bm msgRoute
  264. if err := json.NewDecoder(res).Decode(&bm); err != nil {
  265. t.Fatal(err)
  266. }
  267. mustBeOk(t, bm.Ok)
  268. mustBeNamedRouteOf(t, bm.Route, am.Route.Name, "http://b.com/")
  269. }
  270. func getInPages(e *env, params url.Values) ([][]*routeWithName, error) {
  271. var pages [][]*routeWithName
  272. for {
  273. res, err := e.get("/api/urls/?" + params.Encode())
  274. if err != nil {
  275. return nil, err
  276. }
  277. if res.status != http.StatusOK {
  278. return nil, fmt.Errorf("HTTP status: %d", res.status)
  279. }
  280. var m msgRoutes
  281. if err := json.NewDecoder(res).Decode(&m); err != nil {
  282. return nil, err
  283. }
  284. if !m.Ok {
  285. return nil, errors.New("response is not ok")
  286. }
  287. pages = append(pages, m.Routes)
  288. if m.Next == "" {
  289. return pages, nil
  290. }
  291. params.Set("cursor", m.Next)
  292. }
  293. }
  294. type listTest struct {
  295. Params url.Values
  296. Pages [][]*routeWithName
  297. }
  298. func TestAPIList(t *testing.T) {
  299. e := needEnv(t)
  300. defer e.destroy()
  301. rts := []*routeWithName{
  302. &routeWithName{
  303. Name: "0",
  304. Route: &internal.Route{
  305. URL: "http://0.com/",
  306. Time: time.Now(),
  307. },
  308. },
  309. &routeWithName{
  310. Name: "1",
  311. Route: &internal.Route{
  312. URL: "http://1.com/",
  313. Time: time.Now(),
  314. },
  315. },
  316. &routeWithName{
  317. Name: ":a",
  318. Route: &internal.Route{
  319. URL: "http://ga.com/",
  320. Time: time.Now(),
  321. },
  322. },
  323. &routeWithName{
  324. Name: ":b",
  325. Route: &internal.Route{
  326. URL: "http://gb.com/",
  327. Time: time.Now(),
  328. },
  329. },
  330. &routeWithName{
  331. Name: "a",
  332. Route: &internal.Route{
  333. URL: "http://a.com/",
  334. Time: time.Now(),
  335. },
  336. },
  337. &routeWithName{
  338. Name: "b",
  339. Route: &internal.Route{
  340. URL: "http://b.com/",
  341. Time: time.Now(),
  342. },
  343. },
  344. }
  345. ctx, cancel := context.WithTimeout(context.Background(), time.Minute)
  346. defer cancel()
  347. for _, rt := range rts {
  348. if err := e.backend.Put(ctx, rt.Name, rt.Route); err != nil {
  349. t.Fatal(err)
  350. }
  351. }
  352. tests := []*listTest{
  353. &listTest{
  354. Params: url.Values(map[string][]string{}),
  355. Pages: [][]*routeWithName{
  356. []*routeWithName{rts[0], rts[1], rts[4], rts[5]},
  357. },
  358. },
  359. &listTest{
  360. Params: url.Values(map[string][]string{
  361. "include-generated-names": {"true"},
  362. }),
  363. Pages: [][]*routeWithName{rts},
  364. },
  365. &listTest{
  366. Params: url.Values(map[string][]string{
  367. "include-generated-names": {"false"},
  368. }),
  369. Pages: [][]*routeWithName{
  370. []*routeWithName{rts[0], rts[1], rts[4], rts[5]},
  371. },
  372. },
  373. &listTest{
  374. Params: url.Values(map[string][]string{
  375. "limit": {"2"},
  376. }),
  377. Pages: [][]*routeWithName{
  378. []*routeWithName{rts[0], rts[1]},
  379. []*routeWithName{rts[4], rts[5]},
  380. },
  381. },
  382. &listTest{
  383. Params: url.Values(map[string][]string{
  384. "limit": {"2"},
  385. "include-generated-names": {"true"},
  386. }),
  387. Pages: [][]*routeWithName{
  388. []*routeWithName{rts[0], rts[1]},
  389. []*routeWithName{rts[2], rts[3]},
  390. []*routeWithName{rts[4], rts[5]},
  391. },
  392. },
  393. &listTest{
  394. Params: url.Values(map[string][]string{
  395. "limit": {"2"},
  396. "cursor": {base64.URLEncoding.EncodeToString([]byte{':'})},
  397. }),
  398. Pages: [][]*routeWithName{
  399. []*routeWithName{rts[4], rts[5]},
  400. },
  401. },
  402. &listTest{
  403. Params: url.Values(map[string][]string{
  404. "limit": {"3"},
  405. "include-generated-names": {"true"},
  406. "cursor": {base64.URLEncoding.EncodeToString([]byte{':'})},
  407. }),
  408. Pages: [][]*routeWithName{
  409. []*routeWithName{rts[2], rts[3], rts[4]},
  410. []*routeWithName{rts[5]},
  411. },
  412. },
  413. &listTest{
  414. Params: url.Values(map[string][]string{
  415. "limit": {"1"},
  416. }),
  417. Pages: [][]*routeWithName{
  418. []*routeWithName{rts[0]},
  419. []*routeWithName{rts[1]},
  420. []*routeWithName{rts[4]},
  421. []*routeWithName{rts[5]},
  422. },
  423. },
  424. &listTest{
  425. Params: url.Values(map[string][]string{
  426. "cursor": {base64.URLEncoding.EncodeToString([]byte{'z'})},
  427. }),
  428. Pages: [][]*routeWithName{nil},
  429. },
  430. }
  431. for _, test := range tests {
  432. t.Logf("running tests for ?%s", test.Params.Encode())
  433. pages, err := getInPages(e, test.Params)
  434. if err != nil {
  435. t.Fatal(err)
  436. }
  437. if len(pages) != len(test.Pages) {
  438. t.Fatalf("number of pages mismatch %d vs %d", len(pages), len(test.Pages))
  439. }
  440. for i, n := 0, len(pages); i < n; i++ {
  441. page := pages[i]
  442. expected := test.Pages[i]
  443. if len(page) != len(expected) {
  444. t.Fatalf("page %d, length mismatch expected %d got %d", i, len(expected), len(page))
  445. }
  446. for j, m := 0, len(page); j < m; j++ {
  447. mustBeSameNamedRoute(t, page[j], expected[j])
  448. }
  449. }
  450. }
  451. }
  452. func TestBadList(t *testing.T) {
  453. e := needEnv(t)
  454. defer e.destroy()
  455. tests := map[string]int{
  456. url.Values{
  457. "cursor": {"not a cursor"},
  458. }.Encode(): http.StatusBadRequest,
  459. url.Values{
  460. "limit": {"0"},
  461. }.Encode(): http.StatusBadRequest,
  462. url.Values{
  463. "limit": {"not a limit"},
  464. }.Encode(): http.StatusBadRequest,
  465. url.Values{
  466. "limit": {"100000"},
  467. }.Encode(): http.StatusBadRequest,
  468. url.Values{
  469. "include-generated-names": {"butter"},
  470. }.Encode(): http.StatusBadRequest,
  471. }
  472. for params, status := range tests {
  473. res, err := e.get("/api/urls/?" + params)
  474. if err != nil {
  475. t.Fatal(err)
  476. }
  477. mustHaveStatus(t, res, status)
  478. var m msgErr
  479. if err := json.NewDecoder(res).Decode(&m); err != nil {
  480. t.Fatal(err)
  481. }
  482. mustBeErr(t, &m)
  483. }
  484. }