2015-04-30 07:02:13 +02:00
|
|
|
# [Echo](http://labstack.github.io/echo) [![GoDoc](http://img.shields.io/badge/go-documentation-blue.svg?style=flat-square)](http://godoc.org/github.com/labstack/echo) [![Build Status](http://img.shields.io/travis/labstack/echo.svg?style=flat-square)](https://travis-ci.org/labstack/echo) [![Coverage Status](http://img.shields.io/coveralls/labstack/echo.svg?style=flat-square)](https://coveralls.io/r/labstack/echo)
|
2015-04-15 19:20:49 +02:00
|
|
|
Echo is a fast HTTP router (zero memory allocation) and micro web framework in Go.
|
2015-03-12 23:51:39 +02:00
|
|
|
|
2015-04-14 02:42:44 +02:00
|
|
|
## Features
|
2015-04-19 01:47:48 +02:00
|
|
|
|
2015-05-10 07:06:13 +02:00
|
|
|
- Fast :rocket: HTTP router which smartly prioritize routes.
|
2015-03-30 16:43:09 +02:00
|
|
|
- Extensible middleware/handler, supports:
|
2015-03-26 09:42:34 +02:00
|
|
|
- Middleware
|
2015-05-11 06:13:14 +02:00
|
|
|
- `echo.MiddlewareFunc`
|
2015-04-19 06:46:00 +02:00
|
|
|
- `func(echo.HandlerFunc) echo.HandlerFunc`
|
2015-05-12 05:41:28 +02:00
|
|
|
- `echo.HandlerFunc`
|
2015-05-11 06:12:28 +02:00
|
|
|
- `func(*echo.Context) *echo.HTTPError`
|
2015-04-21 08:17:21 +02:00
|
|
|
- `func(http.Handler) http.Handler`
|
2015-03-30 08:35:08 +02:00
|
|
|
- `http.Handler`
|
|
|
|
- `http.HandlerFunc`
|
|
|
|
- `func(http.ResponseWriter, *http.Request)`
|
2015-03-26 09:42:34 +02:00
|
|
|
- Handler
|
2015-04-27 07:41:41 +02:00
|
|
|
- `echo.HandlerFunc`
|
2015-05-06 06:55:49 +02:00
|
|
|
- `func(*echo.Context) *echo.HTTPError`
|
2015-03-26 09:42:34 +02:00
|
|
|
- `http.Handler`
|
2015-03-30 08:35:08 +02:00
|
|
|
- `http.HandlerFunc`
|
|
|
|
- `func(http.ResponseWriter, *http.Request)`
|
2015-04-19 02:02:02 +02:00
|
|
|
- Sub routing with groups.
|
2015-03-30 16:43:09 +02:00
|
|
|
- Handy encoding/decoding functions.
|
2015-03-21 23:59:09 +02:00
|
|
|
- Serve static files, including index.
|
2015-04-20 07:25:53 +02:00
|
|
|
- Centralized HTTP error handling.
|
2015-04-20 07:32:46 +02:00
|
|
|
- Use a customized function to bind request body to a Go type.
|
2015-04-26 01:10:28 +02:00
|
|
|
- Register a view render so you can use any HTML template engine.
|
2015-03-12 23:51:39 +02:00
|
|
|
|
2015-04-16 19:18:35 +02:00
|
|
|
## Benchmark
|
2015-04-19 01:47:48 +02:00
|
|
|
|
2015-04-16 19:18:35 +02:00
|
|
|
Based on [julienschmidt/go-http-routing-benchmark] (https://github.com/vishr/go-http-routing-benchmark), April 1, 2015
|
2015-04-19 01:47:48 +02:00
|
|
|
|
2015-04-16 19:18:35 +02:00
|
|
|
##### [GitHub API](http://developer.github.com/v3)
|
2015-04-19 01:47:48 +02:00
|
|
|
|
2015-04-16 19:18:35 +02:00
|
|
|
> Echo: 42728 ns/op, 0 B/op, 0 allocs/op
|
|
|
|
|
|
|
|
```
|
|
|
|
BenchmarkAce_GithubAll 20000 65328 ns/op 13792 B/op 167 allocs/op
|
|
|
|
BenchmarkBear_GithubAll 10000 241852 ns/op 79952 B/op 943 allocs/op
|
|
|
|
BenchmarkBeego_GithubAll 3000 458234 ns/op 146272 B/op 2092 allocs/op
|
|
|
|
BenchmarkBone_GithubAll 1000 1923508 ns/op 648016 B/op 8119 allocs/op
|
|
|
|
BenchmarkDenco_GithubAll 20000 81294 ns/op 20224 B/op 167 allocs/op
|
|
|
|
BenchmarkEcho_GithubAll 30000 42728 ns/op 0 B/op 0 allocs/op
|
|
|
|
BenchmarkGin_GithubAll 20000 69373 ns/op 13792 B/op 167 allocs/op
|
|
|
|
BenchmarkGocraftWeb_GithubAll 10000 370978 ns/op 133280 B/op 1889 allocs/op
|
|
|
|
BenchmarkGoji_GithubAll 3000 542766 ns/op 56113 B/op 334 allocs/op
|
|
|
|
BenchmarkGoJsonRest_GithubAll 5000 452551 ns/op 135995 B/op 2940 allocs/op
|
|
|
|
BenchmarkGoRestful_GithubAll 200 9500204 ns/op 707604 B/op 7558 allocs/op
|
|
|
|
BenchmarkGorillaMux_GithubAll 200 6770545 ns/op 153137 B/op 1791 allocs/op
|
|
|
|
BenchmarkHttpRouter_GithubAll 30000 56097 ns/op 13792 B/op 167 allocs/op
|
|
|
|
BenchmarkHttpTreeMux_GithubAll 10000 143175 ns/op 56112 B/op 334 allocs/op
|
|
|
|
BenchmarkKocha_GithubAll 10000 147959 ns/op 23304 B/op 843 allocs/op
|
|
|
|
BenchmarkMacaron_GithubAll 2000 724650 ns/op 224960 B/op 2315 allocs/op
|
|
|
|
BenchmarkMartini_GithubAll 100 10926021 ns/op 237953 B/op 2686 allocs/op
|
|
|
|
BenchmarkPat_GithubAll 300 4525114 ns/op 1504101 B/op 32222 allocs/op
|
|
|
|
BenchmarkRevel_GithubAll 2000 1172963 ns/op 345553 B/op 5918 allocs/op
|
|
|
|
BenchmarkRivet_GithubAll 10000 249104 ns/op 84272 B/op 1079 allocs/op
|
|
|
|
BenchmarkTango_GithubAll 300 4012826 ns/op 1368581 B/op 29157 allocs/op
|
|
|
|
BenchmarkTigerTonic_GithubAll 2000 975450 ns/op 241088 B/op 6052 allocs/op
|
|
|
|
BenchmarkTraffic_GithubAll 200 7540377 ns/op 2664762 B/op 22390 allocs/op
|
|
|
|
BenchmarkVulcan_GithubAll 5000 307241 ns/op 19894 B/op 609 allocs/op
|
|
|
|
BenchmarkZeus_GithubAll 2000 752907 ns/op 300688 B/op 2648 allocs/op
|
|
|
|
```
|
|
|
|
|
2015-04-14 02:42:44 +02:00
|
|
|
## Installation
|
2015-04-19 01:47:48 +02:00
|
|
|
|
2015-04-26 01:10:28 +02:00
|
|
|
```sh
|
|
|
|
$ go get github.com/labstack/echo
|
|
|
|
```
|
2015-03-29 01:44:28 +02:00
|
|
|
|
2015-05-18 22:38:35 +02:00
|
|
|
##[Examples](https://github.com/labstack/echo/tree/master/examples)
|
|
|
|
|
|
|
|
###[Hello, World!](https://github.com/labstack/echo/tree/master/examples/hello)
|
2015-04-20 01:00:23 +02:00
|
|
|
|
2015-03-25 20:16:17 +02:00
|
|
|
```go
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
|
2015-03-27 23:21:30 +02:00
|
|
|
"github.com/labstack/echo"
|
2015-04-21 08:24:34 +02:00
|
|
|
mw "github.com/labstack/echo/middleware"
|
2015-04-11 06:48:26 +02:00
|
|
|
)
|
|
|
|
|
2015-04-19 02:02:02 +02:00
|
|
|
// Handler
|
2015-05-06 06:55:49 +02:00
|
|
|
func hello(c *echo.Context) *echo.HTTPError {
|
|
|
|
return c.String(http.StatusOK, "Hello, World!\n")
|
2015-03-25 20:16:17 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
2015-04-26 01:10:28 +02:00
|
|
|
// Echo instance
|
2015-03-27 23:49:45 +02:00
|
|
|
e := echo.New()
|
2015-03-25 20:16:17 +02:00
|
|
|
|
2015-04-19 02:02:02 +02:00
|
|
|
// Middleware
|
2015-05-15 21:29:14 +02:00
|
|
|
e.Use(mw.Logger())
|
2015-05-18 20:33:11 +02:00
|
|
|
e.Use(mw.Recover())
|
2015-03-25 20:16:17 +02:00
|
|
|
|
2015-04-19 02:02:02 +02:00
|
|
|
// Routes
|
|
|
|
e.Get("/", hello)
|
2015-04-02 14:02:52 +02:00
|
|
|
|
2015-03-25 20:16:17 +02:00
|
|
|
// Start server
|
2015-05-18 07:54:29 +02:00
|
|
|
e.Run(":1323")
|
2015-03-25 20:16:17 +02:00
|
|
|
}
|
|
|
|
```
|
2015-04-26 01:10:28 +02:00
|
|
|
|
2015-05-18 22:38:35 +02:00
|
|
|
###[CRUD](https://github.com/labstack/echo/tree/master/examples/crud)
|
|
|
|
|
|
|
|
```go
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
|
|
|
|
"github.com/labstack/echo"
|
|
|
|
mw "github.com/labstack/echo/middleware"
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
user struct {
|
|
|
|
ID int
|
|
|
|
Name string
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
users = map[int]*user{}
|
|
|
|
seq = 1
|
|
|
|
)
|
|
|
|
|
|
|
|
//----------
|
|
|
|
// Handlers
|
|
|
|
//----------
|
|
|
|
|
|
|
|
func createUser(c *echo.Context) *echo.HTTPError {
|
|
|
|
u := &user{
|
|
|
|
ID: seq,
|
|
|
|
}
|
|
|
|
if he := c.Bind(u); he != nil {
|
|
|
|
return he
|
|
|
|
}
|
|
|
|
users[u.ID] = u
|
|
|
|
seq++
|
|
|
|
return c.JSON(http.StatusCreated, u)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getUser(c *echo.Context) *echo.HTTPError {
|
|
|
|
id, _ := strconv.Atoi(c.Param("id"))
|
|
|
|
return c.JSON(http.StatusOK, users[id])
|
|
|
|
}
|
|
|
|
|
|
|
|
func updateUser(c *echo.Context) *echo.HTTPError {
|
|
|
|
u := new(user)
|
|
|
|
if he := c.Bind(u); he != nil {
|
|
|
|
return he
|
|
|
|
}
|
|
|
|
id, _ := strconv.Atoi(c.Param("id"))
|
|
|
|
users[id].Name = u.Name
|
|
|
|
return c.JSON(http.StatusOK, users[id])
|
|
|
|
}
|
|
|
|
|
|
|
|
func deleteUser(c *echo.Context) *echo.HTTPError {
|
|
|
|
id, _ := strconv.Atoi(c.Param("id"))
|
|
|
|
delete(users, id)
|
|
|
|
return c.NoContent(http.StatusNoContent)
|
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
e := echo.New()
|
|
|
|
|
|
|
|
// Middleware
|
|
|
|
e.Use(mw.Logger())
|
|
|
|
e.Use(mw.Recover())
|
|
|
|
|
|
|
|
// Routes
|
|
|
|
e.Post("/users", createUser)
|
|
|
|
e.Get("/users/:id", getUser)
|
|
|
|
e.Patch("/users/:id", updateUser)
|
|
|
|
e.Delete("/users/:id", deleteUser)
|
|
|
|
|
|
|
|
// Start server
|
|
|
|
e.Run(":1323")
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
###[Website](https://github.com/labstack/echo/tree/master/examples/website)
|
|
|
|
|
|
|
|
```go
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"html/template"
|
|
|
|
|
|
|
|
"github.com/labstack/echo"
|
|
|
|
mw "github.com/labstack/echo/middleware"
|
|
|
|
"github.com/rs/cors"
|
|
|
|
"github.com/thoas/stats"
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
// Template provides HTML template rendering
|
|
|
|
Template struct {
|
|
|
|
templates *template.Template
|
|
|
|
}
|
|
|
|
|
|
|
|
user struct {
|
|
|
|
ID string `json:"id"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
users map[string]user
|
|
|
|
)
|
|
|
|
|
|
|
|
// Render HTML
|
|
|
|
func (t *Template) Render(w io.Writer, name string, data interface{}) *echo.HTTPError {
|
|
|
|
if err := t.templates.ExecuteTemplate(w, name, data); err != nil {
|
|
|
|
return &echo.HTTPError{Error: err}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------
|
|
|
|
// Handlers
|
|
|
|
//----------
|
|
|
|
|
|
|
|
func welcome(c *echo.Context) *echo.HTTPError {
|
|
|
|
return c.Render(http.StatusOK, "welcome", "Joe")
|
|
|
|
}
|
|
|
|
|
|
|
|
func createUser(c *echo.Context) *echo.HTTPError {
|
|
|
|
u := new(user)
|
|
|
|
if err := c.Bind(u); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
users[u.ID] = *u
|
|
|
|
return c.JSON(http.StatusCreated, u)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getUsers(c *echo.Context) *echo.HTTPError {
|
|
|
|
return c.JSON(http.StatusOK, users)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getUser(c *echo.Context) *echo.HTTPError {
|
|
|
|
return c.JSON(http.StatusOK, users[c.P(0)])
|
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
e := echo.New()
|
|
|
|
|
|
|
|
// Middleware
|
|
|
|
e.Use(mw.Logger())
|
|
|
|
e.Use(mw.Recover())
|
|
|
|
|
|
|
|
//------------------------
|
|
|
|
// Third-party middleware
|
|
|
|
//------------------------
|
|
|
|
|
|
|
|
// https://github.com/rs/cors
|
|
|
|
e.Use(cors.Default().Handler)
|
|
|
|
|
|
|
|
// https://github.com/thoas/stats
|
|
|
|
s := stats.New()
|
|
|
|
e.Use(s.Handler)
|
|
|
|
// Route
|
|
|
|
e.Get("/stats", func(c *echo.Context) *echo.HTTPError {
|
|
|
|
return c.JSON(http.StatusOK, s.Data())
|
|
|
|
})
|
|
|
|
|
|
|
|
// Serve index file
|
|
|
|
e.Index("public/index.html")
|
|
|
|
|
|
|
|
// Serve favicon
|
|
|
|
e.Favicon("public/favicon.ico")
|
|
|
|
|
|
|
|
// Serve static files
|
|
|
|
e.Static("/scripts/", "public/scripts")
|
|
|
|
|
|
|
|
//--------
|
|
|
|
// Routes
|
|
|
|
//--------
|
|
|
|
|
|
|
|
e.Post("/users", createUser)
|
|
|
|
e.Get("/users", getUsers)
|
|
|
|
e.Get("/users/:id", getUser)
|
|
|
|
|
|
|
|
//-----------
|
|
|
|
// Templates
|
|
|
|
//-----------
|
|
|
|
|
|
|
|
t := &Template{
|
|
|
|
// Cached templates
|
|
|
|
templates: template.Must(template.ParseFiles("public/views/welcome.html")),
|
|
|
|
}
|
|
|
|
e.Renderer(t)
|
|
|
|
e.Get("/welcome", welcome)
|
|
|
|
|
|
|
|
//-------
|
|
|
|
// Group
|
|
|
|
//-------
|
|
|
|
|
|
|
|
// Group with parent middleware
|
|
|
|
a := e.Group("/admin")
|
|
|
|
a.Use(func(c *echo.Context) *echo.HTTPError {
|
|
|
|
// Security middleware
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
a.Get("", func(c *echo.Context) *echo.HTTPError {
|
|
|
|
return c.String(http.StatusOK, "Welcome admin!")
|
|
|
|
})
|
|
|
|
|
|
|
|
// Group with no parent middleware
|
|
|
|
g := e.Group("/files", func(c *echo.Context) *echo.HTTPError {
|
|
|
|
// Security middleware
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
g.Get("", func(c *echo.Context) *echo.HTTPError {
|
|
|
|
return c.String(http.StatusOK, "Your files!")
|
|
|
|
})
|
|
|
|
|
|
|
|
// Start server
|
|
|
|
e.Run(":1323")
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
users = map[string]user{
|
|
|
|
"1": user{
|
|
|
|
ID: "1",
|
|
|
|
Name: "Wreck-It Ralph",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
###[Middleware](https://github.com/labstack/echo/tree/master/examples/middleware)
|
|
|
|
|
|
|
|
```go
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/labstack/echo"
|
|
|
|
mw "github.com/labstack/echo/middleware"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Handler
|
|
|
|
func hello(c *echo.Context) *echo.HTTPError {
|
|
|
|
return c.String(http.StatusOK, "Hello, World!\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
// Echo instance
|
|
|
|
e := echo.New()
|
|
|
|
|
|
|
|
// Debug mode
|
|
|
|
e.Debug(true)
|
|
|
|
|
|
|
|
//------------
|
|
|
|
// Middleware
|
|
|
|
//------------
|
|
|
|
|
|
|
|
// Logger
|
|
|
|
e.Use(mw.Logger())
|
|
|
|
|
|
|
|
// Recover
|
|
|
|
e.Use(mw.Recover())
|
|
|
|
|
|
|
|
// Basic auth
|
|
|
|
e.Use(mw.BasicAuth(func(u, p string) bool {
|
|
|
|
if u == "joe" && p == "secret" {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}))
|
|
|
|
|
|
|
|
//-------
|
|
|
|
// Slash
|
|
|
|
//-------
|
|
|
|
|
|
|
|
e.Use(mw.StripTrailingSlash())
|
|
|
|
|
|
|
|
// or
|
|
|
|
|
|
|
|
// e.Use(mw.RedirectToSlash())
|
|
|
|
|
|
|
|
// Gzip
|
|
|
|
e.Use(mw.Gzip())
|
|
|
|
|
|
|
|
// Routes
|
|
|
|
e.Get("/", hello)
|
|
|
|
|
|
|
|
// Start server
|
|
|
|
e.Run(":1323")
|
|
|
|
}
|
|
|
|
```
|
2015-04-26 01:10:28 +02:00
|
|
|
|
2015-04-26 01:13:13 +02:00
|
|
|
##[Guide](http://labstack.github.io/echo/guide)
|
2015-04-26 01:10:28 +02:00
|
|
|
|
|
|
|
## Contribute
|
|
|
|
|
|
|
|
**Use issues for everything**
|
2015-04-27 07:41:41 +02:00
|
|
|
|
2015-04-26 01:10:28 +02:00
|
|
|
- Report problems
|
|
|
|
- Discuss before sending pull request
|
|
|
|
- Suggest new features
|
|
|
|
- Improve/fix documentation
|
|
|
|
|
|
|
|
## Credits
|
|
|
|
- [Vishal Rana](https://github.com/vishr) - Author
|
|
|
|
- [Nitin Rana](https://github.com/nr17) - Consultant
|
|
|
|
- [Contributors](https://github.com/labstack/echo/graphs/contributors)
|
|
|
|
|
2015-04-14 02:42:44 +02:00
|
|
|
## License
|
2015-04-19 01:47:48 +02:00
|
|
|
|
2015-04-14 02:38:36 +02:00
|
|
|
[MIT](https://github.com/labstack/echo/blob/master/LICENSE)
|