186 lines
7.3 KiB
Go
186 lines
7.3 KiB
Go
|
/*
|
||
|
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
|