benchmarks_test.go 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. // Copyright 2017 Manu Martinez-Almeida. All rights reserved.
  2. // Use of this source code is governed by a MIT style
  3. // license that can be found in the LICENSE file.
  4. package gin
  5. import (
  6. "html/template"
  7. "net/http"
  8. "os"
  9. "testing"
  10. )
  11. func BenchmarkOneRoute(B *testing.B) {
  12. router := New()
  13. router.GET("/ping", func(c *Context) {})
  14. runRequest(B, router, "GET", "/ping")
  15. }
  16. func BenchmarkRecoveryMiddleware(B *testing.B) {
  17. router := New()
  18. router.Use(Recovery())
  19. router.GET("/", func(c *Context) {})
  20. runRequest(B, router, "GET", "/")
  21. }
  22. func BenchmarkLoggerMiddleware(B *testing.B) {
  23. router := New()
  24. router.Use(LoggerWithWriter(newMockWriter()))
  25. router.GET("/", func(c *Context) {})
  26. runRequest(B, router, "GET", "/")
  27. }
  28. func BenchmarkManyHandlers(B *testing.B) {
  29. router := New()
  30. router.Use(Recovery(), LoggerWithWriter(newMockWriter()))
  31. router.Use(func(c *Context) {})
  32. router.Use(func(c *Context) {})
  33. router.GET("/ping", func(c *Context) {})
  34. runRequest(B, router, "GET", "/ping")
  35. }
  36. func Benchmark5Params(B *testing.B) {
  37. DefaultWriter = os.Stdout
  38. router := New()
  39. router.Use(func(c *Context) {})
  40. router.GET("/param/:param1/:params2/:param3/:param4/:param5", func(c *Context) {})
  41. runRequest(B, router, "GET", "/param/path/to/parameter/john/12345")
  42. }
  43. func BenchmarkOneRouteJSON(B *testing.B) {
  44. router := New()
  45. data := struct {
  46. Status string `json:"status"`
  47. }{"ok"}
  48. router.GET("/json", func(c *Context) {
  49. c.JSON(200, data)
  50. })
  51. runRequest(B, router, "GET", "/json")
  52. }
  53. var htmlContentType = []string{"text/html; charset=utf-8"}
  54. func BenchmarkOneRouteHTML(B *testing.B) {
  55. router := New()
  56. t := template.Must(template.New("index").Parse(`
  57. <html><body><h1>{{.}}</h1></body></html>`))
  58. router.SetHTMLTemplate(t)
  59. router.GET("/html", func(c *Context) {
  60. c.HTML(200, "index", "hola")
  61. })
  62. runRequest(B, router, "GET", "/html")
  63. }
  64. func BenchmarkOneRouteSet(B *testing.B) {
  65. router := New()
  66. router.GET("/ping", func(c *Context) {
  67. c.Set("key", "value")
  68. })
  69. runRequest(B, router, "GET", "/ping")
  70. }
  71. func BenchmarkOneRouteString(B *testing.B) {
  72. router := New()
  73. router.GET("/text", func(c *Context) {
  74. c.String(200, "this is a plain text")
  75. })
  76. runRequest(B, router, "GET", "/text")
  77. }
  78. func BenchmarkManyRoutesFist(B *testing.B) {
  79. router := New()
  80. router.Any("/ping", func(c *Context) {})
  81. runRequest(B, router, "GET", "/ping")
  82. }
  83. func BenchmarkManyRoutesLast(B *testing.B) {
  84. router := New()
  85. router.Any("/ping", func(c *Context) {})
  86. runRequest(B, router, "OPTIONS", "/ping")
  87. }
  88. func Benchmark404(B *testing.B) {
  89. router := New()
  90. router.Any("/something", func(c *Context) {})
  91. router.NoRoute(func(c *Context) {})
  92. runRequest(B, router, "GET", "/ping")
  93. }
  94. func Benchmark404Many(B *testing.B) {
  95. router := New()
  96. router.GET("/", func(c *Context) {})
  97. router.GET("/path/to/something", func(c *Context) {})
  98. router.GET("/post/:id", func(c *Context) {})
  99. router.GET("/view/:id", func(c *Context) {})
  100. router.GET("/favicon.ico", func(c *Context) {})
  101. router.GET("/robots.txt", func(c *Context) {})
  102. router.GET("/delete/:id", func(c *Context) {})
  103. router.GET("/user/:id/:mode", func(c *Context) {})
  104. router.NoRoute(func(c *Context) {})
  105. runRequest(B, router, "GET", "/viewfake")
  106. }
  107. type mockWriter struct {
  108. headers http.Header
  109. }
  110. func newMockWriter() *mockWriter {
  111. return &mockWriter{
  112. http.Header{},
  113. }
  114. }
  115. func (m *mockWriter) Header() (h http.Header) {
  116. return m.headers
  117. }
  118. func (m *mockWriter) Write(p []byte) (n int, err error) {
  119. return len(p), nil
  120. }
  121. func (m *mockWriter) WriteString(s string) (n int, err error) {
  122. return len(s), nil
  123. }
  124. func (m *mockWriter) WriteHeader(int) {}
  125. func runRequest(B *testing.B, r *Engine, method, path string) {
  126. // create fake request
  127. req, err := http.NewRequest(method, path, nil)
  128. if err != nil {
  129. panic(err)
  130. }
  131. w := newMockWriter()
  132. B.ReportAllocs()
  133. B.ResetTimer()
  134. for i := 0; i < B.N; i++ {
  135. r.ServeHTTP(w, req)
  136. }
  137. }