123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185 |
- /*
- Package restful, a lean package for creating REST-style WebServices without magic.
- WebServices and Routes
- A WebService has a collection of Route objects that dispatch incoming Http Requests to a function calls.
- Typically, a WebService has a root path (e.g. /users) and defines common MIME types for its routes.
- WebServices must be added to a container (see below) in order to handler Http requests from a server.
- A Route is defined by a HTTP method, an URL path and (optionally) the MIME types it consumes (Content-Type) and produces (Accept).
- This package has the logic to find the best matching Route and if found, call its Function.
- ws := new(restful.WebService)
- ws.
- Path("/users").
- Consumes(restful.MIME_JSON, restful.MIME_XML).
- Produces(restful.MIME_JSON, restful.MIME_XML)
- ws.Route(ws.GET("/{user-id}").To(u.findUser)) // u is a UserResource
- ...
- // GET http://localhost:8080/users/1
- func (u UserResource) findUser(request *restful.Request, response *restful.Response) {
- id := request.PathParameter("user-id")
- ...
- }
- The (*Request, *Response) arguments provide functions for reading information from the request and writing information back to the response.
- See the example https://github.com/emicklei/go-restful/blob/master/examples/restful-user-resource.go with a full implementation.
- Regular expression matching Routes
- A Route parameter can be specified using the format "uri/{var[:regexp]}" or the special version "uri/{var:*}" for matching the tail of the path.
- For example, /persons/{name:[A-Z][A-Z]} can be used to restrict values for the parameter "name" to only contain capital alphabetic characters.
- Regular expressions must use the standard Go syntax as described in the regexp package. (https://code.google.com/p/re2/wiki/Syntax)
- This feature requires the use of a CurlyRouter.
- Containers
- A Container holds a collection of WebServices, Filters and a http.ServeMux for multiplexing http requests.
- Using the statements "restful.Add(...) and restful.Filter(...)" will register WebServices and Filters to the Default Container.
- The Default container of go-restful uses the http.DefaultServeMux.
- You can create your own Container and create a new http.Server for that particular container.
- container := restful.NewContainer()
- server := &http.Server{Addr: ":8081", Handler: container}
- Filters
- A filter dynamically intercepts requests and responses to transform or use the information contained in the requests or responses.
- You can use filters to perform generic logging, measurement, authentication, redirect, set response headers etc.
- In the restful package there are three hooks into the request,response flow where filters can be added.
- Each filter must define a FilterFunction:
- func (req *restful.Request, resp *restful.Response, chain *restful.FilterChain)
- Use the following statement to pass the request,response pair to the next filter or RouteFunction
- chain.ProcessFilter(req, resp)
- Container Filters
- These are processed before any registered WebService.
- // install a (global) filter for the default container (processed before any webservice)
- restful.Filter(globalLogging)
- WebService Filters
- These are processed before any Route of a WebService.
- // install a webservice filter (processed before any route)
- ws.Filter(webserviceLogging).Filter(measureTime)
- Route Filters
- These are processed before calling the function associated with the Route.
- // install 2 chained route filters (processed before calling findUser)
- ws.Route(ws.GET("/{user-id}").Filter(routeLogging).Filter(NewCountFilter().routeCounter).To(findUser))
- See the example https://github.com/emicklei/go-restful/blob/master/examples/restful-filters.go with full implementations.
- Response Encoding
- Two encodings are supported: gzip and deflate. To enable this for all responses:
- restful.DefaultContainer.EnableContentEncoding(true)
- If a Http request includes the Accept-Encoding header then the response content will be compressed using the specified encoding.
- Alternatively, you can create a Filter that performs the encoding and install it per WebService or Route.
- See the example https://github.com/emicklei/go-restful/blob/master/examples/restful-encoding-filter.go
- OPTIONS support
- By installing a pre-defined container filter, your Webservice(s) can respond to the OPTIONS Http request.
- Filter(OPTIONSFilter())
- CORS
- By installing the filter of a CrossOriginResourceSharing (CORS), your WebService(s) can handle CORS requests.
- cors := CrossOriginResourceSharing{ExposeHeaders: []string{"X-My-Header"}, CookiesAllowed: false, Container: DefaultContainer}
- Filter(cors.Filter)
- Error Handling
- Unexpected things happen. If a request cannot be processed because of a failure, your service needs to tell via the response what happened and why.
- For this reason HTTP status codes exist and it is important to use the correct code in every exceptional situation.
- 400: Bad Request
- If path or query parameters are not valid (content or type) then use http.StatusBadRequest.
- 404: Not Found
- Despite a valid URI, the resource requested may not be available
- 500: Internal Server Error
- If the application logic could not process the request (or write the response) then use http.StatusInternalServerError.
- 405: Method Not Allowed
- The request has a valid URL but the method (GET,PUT,POST,...) is not allowed.
- 406: Not Acceptable
- The request does not have or has an unknown Accept Header set for this operation.
- 415: Unsupported Media Type
- The request does not have or has an unknown Content-Type Header set for this operation.
- ServiceError
- In addition to setting the correct (error) Http status code, you can choose to write a ServiceError message on the response.
- Performance options
- This package has several options that affect the performance of your service. It is important to understand them and how you can change it.
- restful.DefaultContainer.DoNotRecover(false)
- DoNotRecover controls whether panics will be caught to return HTTP 500.
- If set to false, the container will recover from panics.
- Default value is true
- restful.SetCompressorProvider(NewBoundedCachedCompressors(20, 20))
- If content encoding is enabled then the default strategy for getting new gzip/zlib writers and readers is to use a sync.Pool.
- Because writers are expensive structures, performance is even more improved when using a preloaded cache. You can also inject your own implementation.
- Trouble shooting
- This package has the means to produce detail logging of the complete Http request matching process and filter invocation.
- Enabling this feature requires you to set an implementation of restful.StdLogger (e.g. log.Logger) instance such as:
- restful.TraceLogger(log.New(os.Stdout, "[restful] ", log.LstdFlags|log.Lshortfile))
- Logging
- The restful.SetLogger() method allows you to override the logger used by the package. By default restful
- uses the standard library `log` package and logs to stdout. Different logging packages are supported as
- long as they conform to `StdLogger` interface defined in the `log` sub-package, writing an adapter for your
- preferred package is simple.
- Resources
- [project]: https://github.com/emicklei/go-restful
- [examples]: https://github.com/emicklei/go-restful/blob/master/examples
- [design]: http://ernestmicklei.com/2012/11/11/go-restful-api-design/
- [showcases]: https://github.com/emicklei/mora, https://github.com/emicklei/landskape
- (c) 2012-2015, http://ernestmicklei.com. MIT License
- */
- package restful
|